| Line | Branch | Exec | Source |
|---|---|---|---|
| 1 | /* | ||
| 2 | * Copyright (c) 2000 John Walker | ||
| 3 | * Copyright (c) 2016 Paul B Mahol | ||
| 4 | * | ||
| 5 | * This file is part of FFmpeg. | ||
| 6 | * | ||
| 7 | * FFmpeg is free software; you can redistribute it and/or | ||
| 8 | * modify it under the terms of the GNU Lesser General Public | ||
| 9 | * License as published by the Free Software Foundation; either | ||
| 10 | * version 2.1 of the License, or (at your option) any later version. | ||
| 11 | * | ||
| 12 | * FFmpeg is distributed in the hope that it will be useful, | ||
| 13 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
| 14 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | ||
| 15 | * Lesser General Public License for more details. | ||
| 16 | * | ||
| 17 | * You should have received a copy of the GNU Lesser General Public | ||
| 18 | * License along with FFmpeg; if not, write to the Free Software | ||
| 19 | * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA | ||
| 20 | */ | ||
| 21 | |||
| 22 | #include "libavutil/avassert.h" | ||
| 23 | #include "libavutil/intreadwrite.h" | ||
| 24 | #include "libavutil/opt.h" | ||
| 25 | #include "libavutil/parseutils.h" | ||
| 26 | #include "libavutil/pixdesc.h" | ||
| 27 | #include "avfilter.h" | ||
| 28 | #include "filters.h" | ||
| 29 | #include "formats.h" | ||
| 30 | #include "video.h" | ||
| 31 | |||
| 32 | enum CieSystem { | ||
| 33 | XYY, | ||
| 34 | UCS, | ||
| 35 | LUV, | ||
| 36 | NB_CIE | ||
| 37 | }; | ||
| 38 | |||
| 39 | enum ColorsSystems { | ||
| 40 | NTSCsystem, | ||
| 41 | EBUsystem, | ||
| 42 | SMPTEsystem, | ||
| 43 | SMPTE240Msystem, | ||
| 44 | APPLEsystem, | ||
| 45 | wRGBsystem, | ||
| 46 | CIE1931system, | ||
| 47 | Rec709system, | ||
| 48 | Rec2020system, | ||
| 49 | DCIP3, | ||
| 50 | NB_CS | ||
| 51 | }; | ||
| 52 | |||
| 53 | typedef struct CiescopeContext { | ||
| 54 | const AVClass *class; | ||
| 55 | int color_system; | ||
| 56 | unsigned gamuts; | ||
| 57 | int size; | ||
| 58 | int show_white; | ||
| 59 | int correct_gamma; | ||
| 60 | int cie; | ||
| 61 | float intensity; | ||
| 62 | float contrast; | ||
| 63 | int background; | ||
| 64 | int fill; | ||
| 65 | |||
| 66 | float log2lin[65536]; | ||
| 67 | float igamma; | ||
| 68 | float i[3][3]; | ||
| 69 | float m[3][3]; | ||
| 70 | AVFrame *f; | ||
| 71 | void (*filter)(AVFilterContext *ctx, const uint8_t *ptr, | ||
| 72 | ptrdiff_t linesize, | ||
| 73 | float *cx, float *cy, int x, int y); | ||
| 74 | } CiescopeContext; | ||
| 75 | |||
| 76 | #define OFFSET(x) offsetof(CiescopeContext, x) | ||
| 77 | #define FLAGS AV_OPT_FLAG_FILTERING_PARAM|AV_OPT_FLAG_VIDEO_PARAM | ||
| 78 | |||
| 79 | static const AVOption ciescope_options[] = { | ||
| 80 | { "system", "set color system", OFFSET(color_system), AV_OPT_TYPE_INT, {.i64=Rec709system}, 0, NB_CS-1, FLAGS, .unit = "system" }, | ||
| 81 | { "ntsc", "NTSC 1953 Y'I'O' (ITU-R BT.470 System M)", 0, AV_OPT_TYPE_CONST, {.i64=NTSCsystem}, 0, 0, FLAGS, .unit = "system" }, | ||
| 82 | { "470m", "NTSC 1953 Y'I'O' (ITU-R BT.470 System M)", 0, AV_OPT_TYPE_CONST, {.i64=NTSCsystem}, 0, 0, FLAGS, .unit = "system" }, | ||
| 83 | { "ebu", "EBU Y'U'V' (PAL/SECAM) (ITU-R BT.470 System B, G)", 0, AV_OPT_TYPE_CONST, {.i64=EBUsystem}, 0, 0, FLAGS, .unit = "system" }, | ||
| 84 | { "470bg", "EBU Y'U'V' (PAL/SECAM) (ITU-R BT.470 System B, G)", 0, AV_OPT_TYPE_CONST, {.i64=EBUsystem}, 0, 0, FLAGS, .unit = "system" }, | ||
| 85 | { "smpte", "SMPTE-C RGB", 0, AV_OPT_TYPE_CONST, {.i64=SMPTEsystem}, 0, 0, FLAGS, .unit = "system" }, | ||
| 86 | { "240m", "SMPTE-240M Y'PbPr", 0, AV_OPT_TYPE_CONST, {.i64=SMPTE240Msystem},0, 0, FLAGS, .unit = "system" }, | ||
| 87 | { "apple", "Apple RGB", 0, AV_OPT_TYPE_CONST, {.i64=APPLEsystem}, 0, 0, FLAGS, .unit = "system" }, | ||
| 88 | { "widergb", "Adobe Wide Gamut RGB", 0, AV_OPT_TYPE_CONST, {.i64=wRGBsystem}, 0, 0, FLAGS, .unit = "system" }, | ||
| 89 | { "cie1931", "CIE 1931 RGB", 0, AV_OPT_TYPE_CONST, {.i64=CIE1931system}, 0, 0, FLAGS, .unit = "system" }, | ||
| 90 | { "hdtv", "ITU.BT-709 Y'CbCr", 0, AV_OPT_TYPE_CONST, {.i64=Rec709system}, 0, 0, FLAGS, .unit = "system" }, | ||
| 91 | { "rec709", "ITU.BT-709 Y'CbCr", 0, AV_OPT_TYPE_CONST, {.i64=Rec709system}, 0, 0, FLAGS, .unit = "system" }, | ||
| 92 | { "uhdtv", "ITU-R.BT-2020", 0, AV_OPT_TYPE_CONST, {.i64=Rec2020system}, 0, 0, FLAGS, .unit = "system" }, | ||
| 93 | { "rec2020", "ITU-R.BT-2020", 0, AV_OPT_TYPE_CONST, {.i64=Rec2020system}, 0, 0, FLAGS, .unit = "system" }, | ||
| 94 | { "dcip3", "DCI-P3", 0, AV_OPT_TYPE_CONST, {.i64=DCIP3}, 0, 0, FLAGS, .unit = "system" }, | ||
| 95 | { "cie", "set cie system", OFFSET(cie), AV_OPT_TYPE_INT, {.i64=XYY}, 0, NB_CIE-1, FLAGS, .unit = "cie" }, | ||
| 96 | { "xyy", "CIE 1931 xyY", 0, AV_OPT_TYPE_CONST, {.i64=XYY}, 0, 0, FLAGS, .unit = "cie" }, | ||
| 97 | { "ucs", "CIE 1960 UCS", 0, AV_OPT_TYPE_CONST, {.i64=UCS}, 0, 0, FLAGS, .unit = "cie" }, | ||
| 98 | { "luv", "CIE 1976 Luv", 0, AV_OPT_TYPE_CONST, {.i64=LUV}, 0, 0, FLAGS, .unit = "cie" }, | ||
| 99 | { "gamuts", "set what gamuts to draw", OFFSET(gamuts), AV_OPT_TYPE_FLAGS, {.i64=0}, 0, 0xFFF, FLAGS, .unit = "gamuts" }, | ||
| 100 | { "ntsc", NULL, 0, AV_OPT_TYPE_CONST, {.i64=1<<NTSCsystem}, 0, 0, FLAGS, .unit = "gamuts" }, | ||
| 101 | { "470m", NULL, 0, AV_OPT_TYPE_CONST, {.i64=1<<NTSCsystem}, 0, 0, FLAGS, .unit = "gamuts" }, | ||
| 102 | { "ebu", NULL, 0, AV_OPT_TYPE_CONST, {.i64=1<<EBUsystem}, 0, 0, FLAGS, .unit = "gamuts" }, | ||
| 103 | { "470bg", NULL, 0, AV_OPT_TYPE_CONST, {.i64=1<<EBUsystem}, 0, 0, FLAGS, .unit = "gamuts" }, | ||
| 104 | { "smpte", NULL, 0, AV_OPT_TYPE_CONST, {.i64=1<<SMPTEsystem}, 0, 0, FLAGS, .unit = "gamuts" }, | ||
| 105 | { "240m", NULL, 0, AV_OPT_TYPE_CONST, {.i64=1<<SMPTE240Msystem}, 0, 0, FLAGS, .unit = "gamuts" }, | ||
| 106 | { "apple", NULL, 0, AV_OPT_TYPE_CONST, {.i64=1<<APPLEsystem}, 0, 0, FLAGS, .unit = "gamuts" }, | ||
| 107 | { "widergb", NULL, 0, AV_OPT_TYPE_CONST, {.i64=1<<wRGBsystem}, 0, 0, FLAGS, .unit = "gamuts" }, | ||
| 108 | { "cie1931", NULL, 0, AV_OPT_TYPE_CONST, {.i64=1<<CIE1931system}, 0, 0, FLAGS, .unit = "gamuts" }, | ||
| 109 | { "hdtv", NULL, 0, AV_OPT_TYPE_CONST, {.i64=1<<Rec709system}, 0, 0, FLAGS, .unit = "gamuts" }, | ||
| 110 | { "rec709", NULL, 0, AV_OPT_TYPE_CONST, {.i64=1<<Rec709system}, 0, 0, FLAGS, .unit = "gamuts" }, | ||
| 111 | { "uhdtv", NULL, 0, AV_OPT_TYPE_CONST, {.i64=1<<Rec2020system}, 0, 0, FLAGS, .unit = "gamuts" }, | ||
| 112 | { "rec2020", NULL, 0, AV_OPT_TYPE_CONST, {.i64=1<<Rec2020system}, 0, 0, FLAGS, .unit = "gamuts" }, | ||
| 113 | { "dcip3", NULL, 0, AV_OPT_TYPE_CONST, {.i64=1<<DCIP3}, 0, 0, FLAGS, .unit = "gamuts" }, | ||
| 114 | { "size", "set ciescope size", OFFSET(size), AV_OPT_TYPE_INT, {.i64=512}, 256, 8192, FLAGS }, | ||
| 115 | { "s", "set ciescope size", OFFSET(size), AV_OPT_TYPE_INT, {.i64=512}, 256, 8192, FLAGS }, | ||
| 116 | { "intensity", "set ciescope intensity", OFFSET(intensity), AV_OPT_TYPE_FLOAT, {.dbl=0.001}, 0, 1, FLAGS }, | ||
| 117 | { "i", "set ciescope intensity", OFFSET(intensity), AV_OPT_TYPE_FLOAT, {.dbl=0.001}, 0, 1, FLAGS }, | ||
| 118 | { "contrast", NULL, OFFSET(contrast), AV_OPT_TYPE_FLOAT, {.dbl=0.75}, 0, 1, FLAGS }, | ||
| 119 | { "corrgamma", NULL, OFFSET(correct_gamma), AV_OPT_TYPE_BOOL, {.i64=1}, 0, 1, FLAGS }, | ||
| 120 | { "showwhite", NULL, OFFSET(show_white), AV_OPT_TYPE_BOOL, {.i64=0}, 0, 1, FLAGS }, | ||
| 121 | { "gamma", NULL, OFFSET(igamma), AV_OPT_TYPE_DOUBLE, {.dbl=2.6}, 0.1, 6, FLAGS }, | ||
| 122 | { "fill", "fill with CIE colors", OFFSET(fill), AV_OPT_TYPE_BOOL, {.i64=1}, 0, 1, FLAGS }, | ||
| 123 | { NULL } | ||
| 124 | }; | ||
| 125 | |||
| 126 | AVFILTER_DEFINE_CLASS(ciescope); | ||
| 127 | |||
| 128 | static const enum AVPixelFormat in_pix_fmts[] = { | ||
| 129 | AV_PIX_FMT_RGB24, | ||
| 130 | AV_PIX_FMT_RGBA, | ||
| 131 | AV_PIX_FMT_RGB48, | ||
| 132 | AV_PIX_FMT_RGBA64, | ||
| 133 | AV_PIX_FMT_XYZ12, | ||
| 134 | AV_PIX_FMT_NONE | ||
| 135 | }; | ||
| 136 | |||
| 137 | static const enum AVPixelFormat out_pix_fmts[] = { | ||
| 138 | AV_PIX_FMT_RGBA64, | ||
| 139 | AV_PIX_FMT_NONE | ||
| 140 | }; | ||
| 141 | |||
| 142 | ✗ | static int query_formats(const AVFilterContext *ctx, | |
| 143 | AVFilterFormatsConfig **cfg_in, | ||
| 144 | AVFilterFormatsConfig **cfg_out) | ||
| 145 | { | ||
| 146 | int ret; | ||
| 147 | |||
| 148 | ✗ | if ((ret = ff_formats_ref(ff_make_pixel_format_list(in_pix_fmts), &cfg_in[0]->formats)) < 0) | |
| 149 | ✗ | return ret; | |
| 150 | |||
| 151 | ✗ | if ((ret = ff_formats_ref(ff_make_pixel_format_list(out_pix_fmts), &cfg_out[0]->formats)) < 0) | |
| 152 | ✗ | return ret; | |
| 153 | |||
| 154 | ✗ | return 0; | |
| 155 | } | ||
| 156 | |||
| 157 | ✗ | static int config_output(AVFilterLink *outlink) | |
| 158 | { | ||
| 159 | ✗ | CiescopeContext *s = outlink->src->priv; | |
| 160 | |||
| 161 | ✗ | outlink->h = outlink->w = s->size; | |
| 162 | ✗ | outlink->sample_aspect_ratio = (AVRational){1,1}; | |
| 163 | |||
| 164 | ✗ | return 0; | |
| 165 | } | ||
| 166 | |||
| 167 | /* A color system is defined by the CIE x and y coordinates of its | ||
| 168 | three primary illuminants and the x and y coordinates of the white | ||
| 169 | point. */ | ||
| 170 | |||
| 171 | struct ColorSystem { | ||
| 172 | float xRed, yRed, /* Red primary illuminant */ | ||
| 173 | xGreen, yGreen, /* Green primary illuminant */ | ||
| 174 | xBlue, yBlue, /* Blue primary illuminant */ | ||
| 175 | xWhite, yWhite, /* White point */ | ||
| 176 | gamma; /* gamma of nonlinear correction */ | ||
| 177 | }; | ||
| 178 | |||
| 179 | static float const spectral_chromaticity[][3] = { | ||
| 180 | { 0.175560, 0.005294, 0.819146 }, | ||
| 181 | { 0.175483, 0.005286, 0.819231 }, | ||
| 182 | { 0.175400, 0.005279, 0.819321 }, | ||
| 183 | { 0.175317, 0.005271, 0.819412 }, | ||
| 184 | { 0.175237, 0.005263, 0.819500 }, | ||
| 185 | { 0.175161, 0.005256, 0.819582 }, | ||
| 186 | { 0.175088, 0.005247, 0.819665 }, | ||
| 187 | { 0.175015, 0.005236, 0.819749 }, | ||
| 188 | { 0.174945, 0.005226, 0.819829 }, | ||
| 189 | { 0.174880, 0.005221, 0.819899 }, | ||
| 190 | { 0.174821, 0.005221, 0.819959 }, | ||
| 191 | { 0.174770, 0.005229, 0.820001 }, | ||
| 192 | { 0.174722, 0.005238, 0.820040 }, | ||
| 193 | { 0.174665, 0.005236, 0.820098 }, | ||
| 194 | { 0.174595, 0.005218, 0.820187 }, | ||
| 195 | { 0.174510, 0.005182, 0.820309 }, | ||
| 196 | { 0.174409, 0.005127, 0.820464 }, | ||
| 197 | { 0.174308, 0.005068, 0.820624 }, | ||
| 198 | { 0.174222, 0.005017, 0.820761 }, | ||
| 199 | { 0.174156, 0.004981, 0.820863 }, | ||
| 200 | { 0.174112, 0.004964, 0.820924 }, | ||
| 201 | { 0.174088, 0.004964, 0.820948 }, | ||
| 202 | { 0.174073, 0.004973, 0.820955 }, | ||
| 203 | { 0.174057, 0.004982, 0.820961 }, | ||
| 204 | { 0.174036, 0.004986, 0.820978 }, | ||
| 205 | { 0.174008, 0.004981, 0.821012 }, | ||
| 206 | { 0.173972, 0.004964, 0.821064 }, | ||
| 207 | { 0.173932, 0.004943, 0.821125 }, | ||
| 208 | { 0.173889, 0.004926, 0.821185 }, | ||
| 209 | { 0.173845, 0.004916, 0.821239 }, | ||
| 210 | { 0.173801, 0.004915, 0.821284 }, | ||
| 211 | { 0.173754, 0.004925, 0.821321 }, | ||
| 212 | { 0.173705, 0.004937, 0.821358 }, | ||
| 213 | { 0.173655, 0.004944, 0.821401 }, | ||
| 214 | { 0.173606, 0.004940, 0.821454 }, | ||
| 215 | { 0.173560, 0.004923, 0.821517 }, | ||
| 216 | { 0.173514, 0.004895, 0.821590 }, | ||
| 217 | { 0.173468, 0.004865, 0.821667 }, | ||
| 218 | { 0.173424, 0.004836, 0.821740 }, | ||
| 219 | { 0.173380, 0.004813, 0.821807 }, | ||
| 220 | { 0.173337, 0.004797, 0.821866 }, | ||
| 221 | { 0.173291, 0.004786, 0.821923 }, | ||
| 222 | { 0.173238, 0.004779, 0.821983 }, | ||
| 223 | { 0.173174, 0.004775, 0.822051 }, | ||
| 224 | { 0.173101, 0.004774, 0.822125 }, | ||
| 225 | { 0.173021, 0.004775, 0.822204 }, | ||
| 226 | { 0.172934, 0.004781, 0.822285 }, | ||
| 227 | { 0.172843, 0.004791, 0.822366 }, | ||
| 228 | { 0.172751, 0.004799, 0.822450 }, | ||
| 229 | { 0.172662, 0.004802, 0.822536 }, | ||
| 230 | { 0.172577, 0.004799, 0.822624 }, | ||
| 231 | { 0.172489, 0.004795, 0.822715 }, | ||
| 232 | { 0.172396, 0.004796, 0.822808 }, | ||
| 233 | { 0.172296, 0.004803, 0.822901 }, | ||
| 234 | { 0.172192, 0.004815, 0.822993 }, | ||
| 235 | { 0.172087, 0.004833, 0.823081 }, | ||
| 236 | { 0.171982, 0.004855, 0.823163 }, | ||
| 237 | { 0.171871, 0.004889, 0.823240 }, | ||
| 238 | { 0.171741, 0.004939, 0.823319 }, | ||
| 239 | { 0.171587, 0.005010, 0.823402 }, | ||
| 240 | { 0.171407, 0.005102, 0.823490 }, | ||
| 241 | { 0.171206, 0.005211, 0.823583 }, | ||
| 242 | { 0.170993, 0.005334, 0.823674 }, | ||
| 243 | { 0.170771, 0.005470, 0.823759 }, | ||
| 244 | { 0.170541, 0.005621, 0.823838 }, | ||
| 245 | { 0.170301, 0.005789, 0.823911 }, | ||
| 246 | { 0.170050, 0.005974, 0.823976 }, | ||
| 247 | { 0.169786, 0.006177, 0.824037 }, | ||
| 248 | { 0.169505, 0.006398, 0.824097 }, | ||
| 249 | { 0.169203, 0.006639, 0.824158 }, | ||
| 250 | { 0.168878, 0.006900, 0.824222 }, | ||
| 251 | { 0.168525, 0.007184, 0.824291 }, | ||
| 252 | { 0.168146, 0.007491, 0.824363 }, | ||
| 253 | { 0.167746, 0.007821, 0.824433 }, | ||
| 254 | { 0.167328, 0.008175, 0.824496 }, | ||
| 255 | { 0.166895, 0.008556, 0.824549 }, | ||
| 256 | { 0.166446, 0.008964, 0.824589 }, | ||
| 257 | { 0.165977, 0.009402, 0.824622 }, | ||
| 258 | { 0.165483, 0.009865, 0.824652 }, | ||
| 259 | { 0.164963, 0.010351, 0.824687 }, | ||
| 260 | { 0.164412, 0.010858, 0.824731 }, | ||
| 261 | { 0.163828, 0.011385, 0.824787 }, | ||
| 262 | { 0.163210, 0.011937, 0.824853 }, | ||
| 263 | { 0.162552, 0.012520, 0.824928 }, | ||
| 264 | { 0.161851, 0.013137, 0.825011 }, | ||
| 265 | { 0.161105, 0.013793, 0.825102 }, | ||
| 266 | { 0.160310, 0.014491, 0.825199 }, | ||
| 267 | { 0.159466, 0.015232, 0.825302 }, | ||
| 268 | { 0.158573, 0.016015, 0.825412 }, | ||
| 269 | { 0.157631, 0.016840, 0.825529 }, | ||
| 270 | { 0.156641, 0.017705, 0.825654 }, | ||
| 271 | { 0.155605, 0.018609, 0.825786 }, | ||
| 272 | { 0.154525, 0.019556, 0.825920 }, | ||
| 273 | { 0.153397, 0.020554, 0.826049 }, | ||
| 274 | { 0.152219, 0.021612, 0.826169 }, | ||
| 275 | { 0.150985, 0.022740, 0.826274 }, | ||
| 276 | { 0.149691, 0.023950, 0.826359 }, | ||
| 277 | { 0.148337, 0.025247, 0.826416 }, | ||
| 278 | { 0.146928, 0.026635, 0.826437 }, | ||
| 279 | { 0.145468, 0.028118, 0.826413 }, | ||
| 280 | { 0.143960, 0.029703, 0.826337 }, | ||
| 281 | { 0.142405, 0.031394, 0.826201 }, | ||
| 282 | { 0.140796, 0.033213, 0.825991 }, | ||
| 283 | { 0.139121, 0.035201, 0.825679 }, | ||
| 284 | { 0.137364, 0.037403, 0.825233 }, | ||
| 285 | { 0.135503, 0.039879, 0.824618 }, | ||
| 286 | { 0.133509, 0.042692, 0.823798 }, | ||
| 287 | { 0.131371, 0.045876, 0.822753 }, | ||
| 288 | { 0.129086, 0.049450, 0.821464 }, | ||
| 289 | { 0.126662, 0.053426, 0.819912 }, | ||
| 290 | { 0.124118, 0.057803, 0.818079 }, | ||
| 291 | { 0.121469, 0.062588, 0.815944 }, | ||
| 292 | { 0.118701, 0.067830, 0.813468 }, | ||
| 293 | { 0.115807, 0.073581, 0.810612 }, | ||
| 294 | { 0.112776, 0.079896, 0.807328 }, | ||
| 295 | { 0.109594, 0.086843, 0.803563 }, | ||
| 296 | { 0.106261, 0.094486, 0.799253 }, | ||
| 297 | { 0.102776, 0.102864, 0.794360 }, | ||
| 298 | { 0.099128, 0.112007, 0.788865 }, | ||
| 299 | { 0.095304, 0.121945, 0.782751 }, | ||
| 300 | { 0.091294, 0.132702, 0.776004 }, | ||
| 301 | { 0.087082, 0.144317, 0.768601 }, | ||
| 302 | { 0.082680, 0.156866, 0.760455 }, | ||
| 303 | { 0.078116, 0.170420, 0.751464 }, | ||
| 304 | { 0.073437, 0.185032, 0.741531 }, | ||
| 305 | { 0.068706, 0.200723, 0.730571 }, | ||
| 306 | { 0.063993, 0.217468, 0.718539 }, | ||
| 307 | { 0.059316, 0.235254, 0.705430 }, | ||
| 308 | { 0.054667, 0.254096, 0.691238 }, | ||
| 309 | { 0.050031, 0.274002, 0.675967 }, | ||
| 310 | { 0.045391, 0.294976, 0.659633 }, | ||
| 311 | { 0.040757, 0.316981, 0.642262 }, | ||
| 312 | { 0.036195, 0.339900, 0.623905 }, | ||
| 313 | { 0.031756, 0.363598, 0.604646 }, | ||
| 314 | { 0.027494, 0.387921, 0.584584 }, | ||
| 315 | { 0.023460, 0.412703, 0.563837 }, | ||
| 316 | { 0.019705, 0.437756, 0.542539 }, | ||
| 317 | { 0.016268, 0.462955, 0.520777 }, | ||
| 318 | { 0.013183, 0.488207, 0.498610 }, | ||
| 319 | { 0.010476, 0.513404, 0.476120 }, | ||
| 320 | { 0.008168, 0.538423, 0.453409 }, | ||
| 321 | { 0.006285, 0.563068, 0.430647 }, | ||
| 322 | { 0.004875, 0.587116, 0.408008 }, | ||
| 323 | { 0.003982, 0.610447, 0.385570 }, | ||
| 324 | { 0.003636, 0.633011, 0.363352 }, | ||
| 325 | { 0.003859, 0.654823, 0.341318 }, | ||
| 326 | { 0.004646, 0.675898, 0.319456 }, | ||
| 327 | { 0.006011, 0.696120, 0.297869 }, | ||
| 328 | { 0.007988, 0.715342, 0.276670 }, | ||
| 329 | { 0.010603, 0.733413, 0.255984 }, | ||
| 330 | { 0.013870, 0.750186, 0.235943 }, | ||
| 331 | { 0.017766, 0.765612, 0.216622 }, | ||
| 332 | { 0.022244, 0.779630, 0.198126 }, | ||
| 333 | { 0.027273, 0.792104, 0.180623 }, | ||
| 334 | { 0.032820, 0.802926, 0.164254 }, | ||
| 335 | { 0.038852, 0.812016, 0.149132 }, | ||
| 336 | { 0.045328, 0.819391, 0.135281 }, | ||
| 337 | { 0.052177, 0.825164, 0.122660 }, | ||
| 338 | { 0.059326, 0.829426, 0.111249 }, | ||
| 339 | { 0.066716, 0.832274, 0.101010 }, | ||
| 340 | { 0.074302, 0.833803, 0.091894 }, | ||
| 341 | { 0.082053, 0.834090, 0.083856 }, | ||
| 342 | { 0.089942, 0.833289, 0.076769 }, | ||
| 343 | { 0.097940, 0.831593, 0.070468 }, | ||
| 344 | { 0.106021, 0.829178, 0.064801 }, | ||
| 345 | { 0.114161, 0.826207, 0.059632 }, | ||
| 346 | { 0.122347, 0.822770, 0.054882 }, | ||
| 347 | { 0.130546, 0.818928, 0.050526 }, | ||
| 348 | { 0.138702, 0.814774, 0.046523 }, | ||
| 349 | { 0.146773, 0.810395, 0.042832 }, | ||
| 350 | { 0.154722, 0.805864, 0.039414 }, | ||
| 351 | { 0.162535, 0.801238, 0.036226 }, | ||
| 352 | { 0.170237, 0.796519, 0.033244 }, | ||
| 353 | { 0.177850, 0.791687, 0.030464 }, | ||
| 354 | { 0.185391, 0.786728, 0.027881 }, | ||
| 355 | { 0.192876, 0.781629, 0.025495 }, | ||
| 356 | { 0.200309, 0.776399, 0.023292 }, | ||
| 357 | { 0.207690, 0.771055, 0.021255 }, | ||
| 358 | { 0.215030, 0.765595, 0.019375 }, | ||
| 359 | { 0.222337, 0.760020, 0.017643 }, | ||
| 360 | { 0.229620, 0.754329, 0.016051 }, | ||
| 361 | { 0.236885, 0.748524, 0.014591 }, | ||
| 362 | { 0.244133, 0.742614, 0.013253 }, | ||
| 363 | { 0.251363, 0.736606, 0.012031 }, | ||
| 364 | { 0.258578, 0.730507, 0.010916 }, | ||
| 365 | { 0.265775, 0.724324, 0.009901 }, | ||
| 366 | { 0.272958, 0.718062, 0.008980 }, | ||
| 367 | { 0.280129, 0.711725, 0.008146 }, | ||
| 368 | { 0.287292, 0.705316, 0.007391 }, | ||
| 369 | { 0.294450, 0.698842, 0.006708 }, | ||
| 370 | { 0.301604, 0.692308, 0.006088 }, | ||
| 371 | { 0.308760, 0.685712, 0.005528 }, | ||
| 372 | { 0.315914, 0.679063, 0.005022 }, | ||
| 373 | { 0.323066, 0.672367, 0.004566 }, | ||
| 374 | { 0.330216, 0.665628, 0.004156 }, | ||
| 375 | { 0.337363, 0.658848, 0.003788 }, | ||
| 376 | { 0.344513, 0.652028, 0.003459 }, | ||
| 377 | { 0.351664, 0.645172, 0.003163 }, | ||
| 378 | { 0.358814, 0.638287, 0.002899 }, | ||
| 379 | { 0.365959, 0.631379, 0.002662 }, | ||
| 380 | { 0.373102, 0.624451, 0.002448 }, | ||
| 381 | { 0.380244, 0.617502, 0.002254 }, | ||
| 382 | { 0.387379, 0.610542, 0.002079 }, | ||
| 383 | { 0.394507, 0.603571, 0.001922 }, | ||
| 384 | { 0.401626, 0.596592, 0.001782 }, | ||
| 385 | { 0.408736, 0.589607, 0.001657 }, | ||
| 386 | { 0.415836, 0.582618, 0.001546 }, | ||
| 387 | { 0.422921, 0.575631, 0.001448 }, | ||
| 388 | { 0.429989, 0.568649, 0.001362 }, | ||
| 389 | { 0.437036, 0.561676, 0.001288 }, | ||
| 390 | { 0.444062, 0.554714, 0.001224 }, | ||
| 391 | { 0.451065, 0.547766, 0.001169 }, | ||
| 392 | { 0.458041, 0.540837, 0.001123 }, | ||
| 393 | { 0.464986, 0.533930, 0.001084 }, | ||
| 394 | { 0.471899, 0.527051, 0.001051 }, | ||
| 395 | { 0.478775, 0.520202, 0.001023 }, | ||
| 396 | { 0.485612, 0.513389, 0.001000 }, | ||
| 397 | { 0.492405, 0.506615, 0.000980 }, | ||
| 398 | { 0.499151, 0.499887, 0.000962 }, | ||
| 399 | { 0.505845, 0.493211, 0.000944 }, | ||
| 400 | { 0.512486, 0.486591, 0.000923 }, | ||
| 401 | { 0.519073, 0.480029, 0.000899 }, | ||
| 402 | { 0.525600, 0.473527, 0.000872 }, | ||
| 403 | { 0.532066, 0.467091, 0.000843 }, | ||
| 404 | { 0.538463, 0.460725, 0.000812 }, | ||
| 405 | { 0.544787, 0.454434, 0.000779 }, | ||
| 406 | { 0.551031, 0.448225, 0.000744 }, | ||
| 407 | { 0.557193, 0.442099, 0.000708 }, | ||
| 408 | { 0.563269, 0.436058, 0.000673 }, | ||
| 409 | { 0.569257, 0.430102, 0.000641 }, | ||
| 410 | { 0.575151, 0.424232, 0.000616 }, | ||
| 411 | { 0.580953, 0.418447, 0.000601 }, | ||
| 412 | { 0.586650, 0.412758, 0.000591 }, | ||
| 413 | { 0.592225, 0.407190, 0.000586 }, | ||
| 414 | { 0.597658, 0.401762, 0.000580 }, | ||
| 415 | { 0.602933, 0.396497, 0.000571 }, | ||
| 416 | { 0.608035, 0.391409, 0.000556 }, | ||
| 417 | { 0.612977, 0.386486, 0.000537 }, | ||
| 418 | { 0.617779, 0.381706, 0.000516 }, | ||
| 419 | { 0.622459, 0.377047, 0.000493 }, | ||
| 420 | { 0.627037, 0.372491, 0.000472 }, | ||
| 421 | { 0.631521, 0.368026, 0.000453 }, | ||
| 422 | { 0.635900, 0.363665, 0.000435 }, | ||
| 423 | { 0.640156, 0.359428, 0.000416 }, | ||
| 424 | { 0.644273, 0.355331, 0.000396 }, | ||
| 425 | { 0.648233, 0.351395, 0.000372 }, | ||
| 426 | { 0.652028, 0.347628, 0.000344 }, | ||
| 427 | { 0.655669, 0.344018, 0.000313 }, | ||
| 428 | { 0.659166, 0.340553, 0.000281 }, | ||
| 429 | { 0.662528, 0.337221, 0.000251 }, | ||
| 430 | { 0.665764, 0.334011, 0.000226 }, | ||
| 431 | { 0.668874, 0.330919, 0.000207 }, | ||
| 432 | { 0.671859, 0.327947, 0.000194 }, | ||
| 433 | { 0.674720, 0.325095, 0.000185 }, | ||
| 434 | { 0.677459, 0.322362, 0.000179 }, | ||
| 435 | { 0.680079, 0.319747, 0.000174 }, | ||
| 436 | { 0.682582, 0.317249, 0.000170 }, | ||
| 437 | { 0.684971, 0.314863, 0.000167 }, | ||
| 438 | { 0.687250, 0.312586, 0.000164 }, | ||
| 439 | { 0.689426, 0.310414, 0.000160 }, | ||
| 440 | { 0.691504, 0.308342, 0.000154 }, | ||
| 441 | { 0.693490, 0.306366, 0.000145 }, | ||
| 442 | { 0.695389, 0.304479, 0.000133 }, | ||
| 443 | { 0.697206, 0.302675, 0.000119 }, | ||
| 444 | { 0.698944, 0.300950, 0.000106 }, | ||
| 445 | { 0.700606, 0.299301, 0.000093 }, | ||
| 446 | { 0.702193, 0.297725, 0.000083 }, | ||
| 447 | { 0.703709, 0.296217, 0.000074 }, | ||
| 448 | { 0.705163, 0.294770, 0.000067 }, | ||
| 449 | { 0.706563, 0.293376, 0.000061 }, | ||
| 450 | { 0.707918, 0.292027, 0.000055 }, | ||
| 451 | { 0.709231, 0.290719, 0.000050 }, | ||
| 452 | { 0.710500, 0.289453, 0.000047 }, | ||
| 453 | { 0.711724, 0.288232, 0.000044 }, | ||
| 454 | { 0.712901, 0.287057, 0.000041 }, | ||
| 455 | { 0.714032, 0.285929, 0.000040 }, | ||
| 456 | { 0.715117, 0.284845, 0.000038 }, | ||
| 457 | { 0.716159, 0.283804, 0.000036 }, | ||
| 458 | { 0.717159, 0.282806, 0.000035 }, | ||
| 459 | { 0.718116, 0.281850, 0.000034 }, | ||
| 460 | { 0.719033, 0.280935, 0.000032 }, | ||
| 461 | { 0.719912, 0.280058, 0.000030 }, | ||
| 462 | { 0.720753, 0.279219, 0.000028 }, | ||
| 463 | { 0.721555, 0.278420, 0.000026 }, | ||
| 464 | { 0.722315, 0.277662, 0.000023 }, | ||
| 465 | { 0.723032, 0.276948, 0.000020 }, | ||
| 466 | { 0.723702, 0.276282, 0.000016 }, | ||
| 467 | { 0.724328, 0.275660, 0.000012 }, | ||
| 468 | { 0.724914, 0.275078, 0.000007 }, | ||
| 469 | { 0.725467, 0.274530, 0.000003 }, | ||
| 470 | { 0.725992, 0.274008, 0.000000 }, | ||
| 471 | { 0.726495, 0.273505, 0.000000 }, | ||
| 472 | { 0.726975, 0.273025, 0.000000 }, | ||
| 473 | { 0.727432, 0.272568, 0.000000 }, | ||
| 474 | { 0.727864, 0.272136, 0.000000 }, | ||
| 475 | { 0.728272, 0.271728, 0.000000 }, | ||
| 476 | { 0.728656, 0.271344, 0.000000 }, | ||
| 477 | { 0.729020, 0.270980, 0.000000 }, | ||
| 478 | { 0.729361, 0.270639, 0.000000 }, | ||
| 479 | { 0.729678, 0.270322, 0.000000 }, | ||
| 480 | { 0.729969, 0.270031, 0.000000 }, | ||
| 481 | { 0.730234, 0.269766, 0.000000 }, | ||
| 482 | { 0.730474, 0.269526, 0.000000 }, | ||
| 483 | { 0.730693, 0.269307, 0.000000 }, | ||
| 484 | { 0.730896, 0.269104, 0.000000 }, | ||
| 485 | { 0.731089, 0.268911, 0.000000 }, | ||
| 486 | { 0.731280, 0.268720, 0.000000 }, | ||
| 487 | { 0.731467, 0.268533, 0.000000 }, | ||
| 488 | { 0.731650, 0.268350, 0.000000 }, | ||
| 489 | { 0.731826, 0.268174, 0.000000 }, | ||
| 490 | { 0.731993, 0.268007, 0.000000 }, | ||
| 491 | { 0.732150, 0.267850, 0.000000 }, | ||
| 492 | { 0.732300, 0.267700, 0.000000 }, | ||
| 493 | { 0.732443, 0.267557, 0.000000 }, | ||
| 494 | { 0.732581, 0.267419, 0.000000 }, | ||
| 495 | { 0.732719, 0.267281, 0.000000 }, | ||
| 496 | { 0.732859, 0.267141, 0.000000 }, | ||
| 497 | { 0.733000, 0.267000, 0.000000 }, | ||
| 498 | { 0.733142, 0.266858, 0.000000 }, | ||
| 499 | { 0.733281, 0.266719, 0.000000 }, | ||
| 500 | { 0.733417, 0.266583, 0.000000 }, | ||
| 501 | { 0.733551, 0.266449, 0.000000 }, | ||
| 502 | { 0.733683, 0.266317, 0.000000 }, | ||
| 503 | { 0.733813, 0.266187, 0.000000 }, | ||
| 504 | { 0.733936, 0.266064, 0.000000 }, | ||
| 505 | { 0.734047, 0.265953, 0.000000 }, | ||
| 506 | { 0.734143, 0.265857, 0.000000 }, | ||
| 507 | { 0.734221, 0.265779, 0.000000 }, | ||
| 508 | { 0.734286, 0.265714, 0.000000 }, | ||
| 509 | { 0.734341, 0.265659, 0.000000 }, | ||
| 510 | { 0.734390, 0.265610, 0.000000 }, | ||
| 511 | { 0.734438, 0.265562, 0.000000 }, | ||
| 512 | { 0.734482, 0.265518, 0.000000 }, | ||
| 513 | { 0.734523, 0.265477, 0.000000 }, | ||
| 514 | { 0.734560, 0.265440, 0.000000 }, | ||
| 515 | { 0.734592, 0.265408, 0.000000 }, | ||
| 516 | { 0.734621, 0.265379, 0.000000 }, | ||
| 517 | { 0.734649, 0.265351, 0.000000 }, | ||
| 518 | { 0.734673, 0.265327, 0.000000 }, | ||
| 519 | { 0.734690, 0.265310, 0.000000 }, | ||
| 520 | { 0.734690, 0.265310, 0.000000 }, | ||
| 521 | { 0.734690, 0.265310, 0.000000 }, | ||
| 522 | { 0.734690, 0.265310, 0.000000 }, | ||
| 523 | { 0.734690, 0.265310, 0.000000 }, | ||
| 524 | { 0.734690, 0.265310, 0.000000 }, | ||
| 525 | { 0.734690, 0.265310, 0.000000 }, | ||
| 526 | { 0.734690, 0.265310, 0.000000 }, | ||
| 527 | { 0.734690, 0.265310, 0.000000 }, | ||
| 528 | { 0.734690, 0.265310, 0.000000 }, | ||
| 529 | { 0.734690, 0.265310, 0.000000 }, | ||
| 530 | { 0.734690, 0.265310, 0.000000 }, | ||
| 531 | { 0.734690, 0.265310, 0.000000 }, | ||
| 532 | { 0.734690, 0.265310, 0.000000 }, | ||
| 533 | { 0.734690, 0.265310, 0.000000 }, | ||
| 534 | { 0.734690, 0.265310, 0.000000 }, | ||
| 535 | { 0.734690, 0.265310, 0.000000 }, | ||
| 536 | { 0.734690, 0.265310, 0.000000 }, | ||
| 537 | { 0.734690, 0.265310, 0.000000 }, | ||
| 538 | { 0.734690, 0.265310, 0.000000 }, | ||
| 539 | { 0.734690, 0.265310, 0.000000 }, | ||
| 540 | { 0.734690, 0.265310, 0.000000 }, | ||
| 541 | { 0.734690, 0.265310, 0.000000 }, | ||
| 542 | { 0.734690, 0.265310, 0.000000 }, | ||
| 543 | { 0.734690, 0.265310, 0.000000 }, | ||
| 544 | { 0.734690, 0.265310, 0.000000 }, | ||
| 545 | { 0.734690, 0.265310, 0.000000 }, | ||
| 546 | { 0.734690, 0.265310, 0.000000 }, | ||
| 547 | { 0.734690, 0.265310, 0.000000 }, | ||
| 548 | { 0.734690, 0.265310, 0.000000 }, | ||
| 549 | { 0.734690, 0.265310, 0.000000 }, | ||
| 550 | { 0.734690, 0.265310, 0.000000 }, | ||
| 551 | { 0.734690, 0.265310, 0.000000 }, | ||
| 552 | { 0.734690, 0.265310, 0.000000 }, | ||
| 553 | { 0.734690, 0.265310, 0.000000 }, | ||
| 554 | { 0.734690, 0.265310, 0.000000 }, | ||
| 555 | { 0.734690, 0.265310, 0.000000 }, | ||
| 556 | { 0.734690, 0.265310, 0.000000 }, | ||
| 557 | { 0.734690, 0.265310, 0.000000 }, | ||
| 558 | { 0.734690, 0.265310, 0.000000 }, | ||
| 559 | { 0.734690, 0.265310, 0.000000 }, | ||
| 560 | { 0.734690, 0.265310, 0.000000 }, | ||
| 561 | { 0.734690, 0.265310, 0.000000 }, | ||
| 562 | { 0.734690, 0.265310, 0.000000 }, | ||
| 563 | { 0.734690, 0.265310, 0.000000 }, | ||
| 564 | { 0.734690, 0.265310, 0.000000 }, | ||
| 565 | { 0.734690, 0.265310, 0.000000 }, | ||
| 566 | { 0.734690, 0.265310, 0.000000 }, | ||
| 567 | { 0.734690, 0.265310, 0.000000 }, | ||
| 568 | { 0.734690, 0.265310, 0.000000 }, | ||
| 569 | { 0.734690, 0.265310, 0.000000 }, | ||
| 570 | { 0.734690, 0.265310, 0.000000 }, | ||
| 571 | { 0.734690, 0.265310, 0.000000 }, | ||
| 572 | { 0.734690, 0.265310, 0.000000 }, | ||
| 573 | { 0.734690, 0.265310, 0.000000 }, | ||
| 574 | { 0.734690, 0.265310, 0.000000 }, | ||
| 575 | { 0.734690, 0.265310, 0.000000 }, | ||
| 576 | { 0.734690, 0.265310, 0.000000 }, | ||
| 577 | { 0.734690, 0.265310, 0.000000 }, | ||
| 578 | { 0.734690, 0.265310, 0.000000 }, | ||
| 579 | { 0.734690, 0.265310, 0.000000 }, | ||
| 580 | { 0.734690, 0.265310, 0.000000 }, | ||
| 581 | { 0.734690, 0.265310, 0.000000 }, | ||
| 582 | { 0.734690, 0.265310, 0.000000 }, | ||
| 583 | { 0.734690, 0.265310, 0.000000 }, | ||
| 584 | { 0.734690, 0.265310, 0.000000 }, | ||
| 585 | { 0.734690, 0.265310, 0.000000 }, | ||
| 586 | { 0.734690, 0.265310, 0.000000 }, | ||
| 587 | { 0.734690, 0.265310, 0.000000 }, | ||
| 588 | { 0.734690, 0.265310, 0.000000 }, | ||
| 589 | { 0.734690, 0.265310, 0.000000 }, | ||
| 590 | { 0.734690, 0.265310, 0.000000 }, | ||
| 591 | { 0.734690, 0.265310, 0.000000 }, | ||
| 592 | { 0.734690, 0.265310, 0.000000 }, | ||
| 593 | { 0.734690, 0.265310, 0.000000 }, | ||
| 594 | { 0.734690, 0.265310, 0.000000 }, | ||
| 595 | { 0.734690, 0.265310, 0.000000 }, | ||
| 596 | { 0.734690, 0.265310, 0.000000 }, | ||
| 597 | { 0.734690, 0.265310, 0.000000 }, | ||
| 598 | { 0.734690, 0.265310, 0.000000 }, | ||
| 599 | { 0.734690, 0.265310, 0.000000 }, | ||
| 600 | { 0.734690, 0.265310, 0.000000 }, | ||
| 601 | { 0.734690, 0.265310, 0.000000 }, | ||
| 602 | { 0.734690, 0.265310, 0.000000 }, | ||
| 603 | { 0.734690, 0.265310, 0.000000 }, | ||
| 604 | { 0.734690, 0.265310, 0.000000 }, | ||
| 605 | { 0.734690, 0.265310, 0.000000 }, | ||
| 606 | { 0.734690, 0.265310, 0.000000 }, | ||
| 607 | { 0.734690, 0.265310, 0.000000 }, | ||
| 608 | { 0.734690, 0.265310, 0.000000 }, | ||
| 609 | { 0.734690, 0.265310, 0.000000 }, | ||
| 610 | { 0.734690, 0.265310, 0.000000 }, | ||
| 611 | { 0.734690, 0.265310, 0.000000 }, | ||
| 612 | { 0.734690, 0.265310, 0.000000 }, | ||
| 613 | { 0.734690, 0.265310, 0.000000 }, | ||
| 614 | { 0.734690, 0.265310, 0.000000 }, | ||
| 615 | { 0.734690, 0.265310, 0.000000 }, | ||
| 616 | { 0.734690, 0.265310, 0.000000 }, | ||
| 617 | { 0.734690, 0.265310, 0.000000 }, | ||
| 618 | { 0.734690, 0.265310, 0.000000 }, | ||
| 619 | { 0.734690, 0.265310, 0.000000 }, | ||
| 620 | { 0.734690, 0.265310, 0.000000 }, | ||
| 621 | { 0.734690, 0.265310, 0.000000 }, | ||
| 622 | { 0.734690, 0.265310, 0.000000 }, | ||
| 623 | { 0.734690, 0.265310, 0.000000 }, | ||
| 624 | { 0.734690, 0.265310, 0.000000 }, | ||
| 625 | { 0.734690, 0.265310, 0.000000 }, | ||
| 626 | { 0.734690, 0.265310, 0.000000 }, | ||
| 627 | { 0.734690, 0.265310, 0.000000 }, | ||
| 628 | { 0.734690, 0.265310, 0.000000 }, | ||
| 629 | { 0.734690, 0.265310, 0.000000 }, | ||
| 630 | { 0.734690, 0.265310, 0.000000 }, | ||
| 631 | { 0.734690, 0.265310, 0.000000 }, | ||
| 632 | { 0.734690, 0.265310, 0.000000 }, | ||
| 633 | { 0.734690, 0.265310, 0.000000 }, | ||
| 634 | { 0.734690, 0.265310, 0.000000 }, | ||
| 635 | { 0.734690, 0.265310, 0.000000 }, | ||
| 636 | { 0.734690, 0.265310, 0.000000 }, | ||
| 637 | { 0.734690, 0.265310, 0.000000 }, | ||
| 638 | { 0.734690, 0.265310, 0.000000 }, | ||
| 639 | { 0.734690, 0.265310, 0.000000 }, | ||
| 640 | { 0.734690, 0.265310, 0.000000 }, | ||
| 641 | { 0.734690, 0.265310, 0.000000 }, | ||
| 642 | { 0.734690, 0.265310, 0.000000 }, | ||
| 643 | { 0.734690, 0.265310, 0.000000 }, | ||
| 644 | { 0.734690, 0.265310, 0.000000 }, | ||
| 645 | { 0.734690, 0.265310, 0.000000 }, | ||
| 646 | { 0.734690, 0.265310, 0.000000 }, | ||
| 647 | { 0.734690, 0.265310, 0.000000 }, | ||
| 648 | { 0.734690, 0.265310, 0.000000 }, | ||
| 649 | { 0.734690, 0.265310, 0.000000 }, | ||
| 650 | { 0.734690, 0.265310, 0.000000 }, | ||
| 651 | }; | ||
| 652 | |||
| 653 | |||
| 654 | /* Standard white point chromaticities. */ | ||
| 655 | |||
| 656 | #define C 0.310063, 0.316158 | ||
| 657 | #define E 1.0/3.0, 1.0/3.0 | ||
| 658 | #define D50 0.34570, 0.3585 | ||
| 659 | #define D65 0.312713, 0.329016 | ||
| 660 | |||
| 661 | /* Gamma of nonlinear correction. | ||
| 662 | See Charles Poynton's ColorFAQ Item 45 and GammaFAQ Item 6 at | ||
| 663 | http://www.inforamp.net/~poynton/ColorFAQ.html | ||
| 664 | http://www.inforamp.net/~poynton/GammaFAQ.html | ||
| 665 | */ | ||
| 666 | |||
| 667 | #define GAMMA_REC709 0. /* Rec. 709 */ | ||
| 668 | |||
| 669 | static const struct ColorSystem color_systems[] = { | ||
| 670 | [NTSCsystem] = { | ||
| 671 | 0.67, 0.33, 0.21, 0.71, 0.14, 0.08, | ||
| 672 | C, GAMMA_REC709 | ||
| 673 | }, | ||
| 674 | [EBUsystem] = { | ||
| 675 | 0.64, 0.33, 0.29, 0.60, 0.15, 0.06, | ||
| 676 | D65, GAMMA_REC709 | ||
| 677 | }, | ||
| 678 | [SMPTEsystem] = { | ||
| 679 | 0.630, 0.340, 0.310, 0.595, 0.155, 0.070, | ||
| 680 | D65, GAMMA_REC709 | ||
| 681 | }, | ||
| 682 | [SMPTE240Msystem] = { | ||
| 683 | 0.670, 0.330, 0.210, 0.710, 0.150, 0.060, | ||
| 684 | D65, GAMMA_REC709 | ||
| 685 | }, | ||
| 686 | [APPLEsystem] = { | ||
| 687 | 0.625, 0.340, 0.280, 0.595, 0.115, 0.070, | ||
| 688 | D65, GAMMA_REC709 | ||
| 689 | }, | ||
| 690 | [wRGBsystem] = { | ||
| 691 | 0.7347, 0.2653, 0.1152, 0.8264, 0.1566, 0.0177, | ||
| 692 | D50, GAMMA_REC709 | ||
| 693 | }, | ||
| 694 | [CIE1931system] = { | ||
| 695 | 0.7347, 0.2653, 0.2738, 0.7174, 0.1666, 0.0089, | ||
| 696 | E, GAMMA_REC709 | ||
| 697 | }, | ||
| 698 | [Rec709system] = { | ||
| 699 | 0.64, 0.33, 0.30, 0.60, 0.15, 0.06, | ||
| 700 | D65, GAMMA_REC709 | ||
| 701 | }, | ||
| 702 | [Rec2020system] = { | ||
| 703 | 0.708, 0.292, 0.170, 0.797, 0.131, 0.046, | ||
| 704 | D65, GAMMA_REC709 | ||
| 705 | }, | ||
| 706 | [DCIP3] = { | ||
| 707 | 0.680, 0.320, 0.265, 0.690, 0.150, 0.060, | ||
| 708 | 0.314, 0.351, GAMMA_REC709 | ||
| 709 | }, | ||
| 710 | }; | ||
| 711 | |||
| 712 | /* | ||
| 713 | static struct ColorSystem CustomSystem = { | ||
| 714 | "Custom", | ||
| 715 | 0.64, 0.33, 0.30, 0.60, 0.15, 0.06, | ||
| 716 | D65, GAMMA_REC709 | ||
| 717 | }; | ||
| 718 | */ | ||
| 719 | |||
| 720 | static void | ||
| 721 | ✗ | uv_to_xy(float const u, | |
| 722 | float const v, | ||
| 723 | float *const xc, | ||
| 724 | float *const yc) | ||
| 725 | { | ||
| 726 | /* | ||
| 727 | Given 1970 coordinates u, v, determine 1931 chromaticities x, y | ||
| 728 | */ | ||
| 729 | ✗ | *xc = 3.f*u / (2.f*u - 8.f*v + 4.f); | |
| 730 | ✗ | *yc = 2.f*v / (2.f*u - 8.f*v + 4.f); | |
| 731 | ✗ | } | |
| 732 | |||
| 733 | static void | ||
| 734 | ✗ | upvp_to_xy(float const up, | |
| 735 | float const vp, | ||
| 736 | float * const xc, | ||
| 737 | float * const yc) | ||
| 738 | { | ||
| 739 | /* | ||
| 740 | Given 1976 coordinates u', v', determine 1931 chromaticities x, y | ||
| 741 | */ | ||
| 742 | ✗ | *xc = 9*up / (6*up - 16*vp + 12); | |
| 743 | ✗ | *yc = 4*vp / (6*up - 16*vp + 12); | |
| 744 | ✗ | } | |
| 745 | |||
| 746 | static void | ||
| 747 | ✗ | xy_to_upvp(float xc, | |
| 748 | float yc, | ||
| 749 | float * const up, | ||
| 750 | float * const vp) | ||
| 751 | { | ||
| 752 | /* | ||
| 753 | Given 1931 chromaticities x, y, determine 1976 coordinates u', v' | ||
| 754 | */ | ||
| 755 | ✗ | const float scale = 1.f / (-2.f*xc + 12.f*yc + 3.f); | |
| 756 | ✗ | *up = 4.f*xc * scale; | |
| 757 | ✗ | *vp = 9.f*yc * scale; | |
| 758 | ✗ | } | |
| 759 | |||
| 760 | static void | ||
| 761 | ✗ | xy_to_uv(float xc, | |
| 762 | float yc, | ||
| 763 | float * const u, | ||
| 764 | float * const v) | ||
| 765 | { | ||
| 766 | /* | ||
| 767 | Given 1931 chromaticities x, y, determine 1960 coordinates u, v | ||
| 768 | */ | ||
| 769 | ✗ | const float scale = 1.f / (-2.f*xc + 12.f*yc + 3.f); | |
| 770 | ✗ | *u = 4.f*xc * scale; | |
| 771 | ✗ | *v = 6.f*yc * scale; | |
| 772 | ✗ | } | |
| 773 | |||
| 774 | static void | ||
| 775 | ✗ | xyz_to_rgb(const float m[3][3], | |
| 776 | float xc, float yc, float zc, | ||
| 777 | float * const r, float * const g, float * const b) | ||
| 778 | { | ||
| 779 | ✗ | *r = m[0][0]*xc + m[0][1]*yc + m[0][2]*zc; | |
| 780 | ✗ | *g = m[1][0]*xc + m[1][1]*yc + m[1][2]*zc; | |
| 781 | ✗ | *b = m[2][0]*xc + m[2][1]*yc + m[2][2]*zc; | |
| 782 | ✗ | } | |
| 783 | |||
| 784 | ✗ | static void invert_matrix3x3(float in[3][3], float out[3][3]) | |
| 785 | { | ||
| 786 | ✗ | float m00 = in[0][0], m01 = in[0][1], m02 = in[0][2], | |
| 787 | ✗ | m10 = in[1][0], m11 = in[1][1], m12 = in[1][2], | |
| 788 | ✗ | m20 = in[2][0], m21 = in[2][1], m22 = in[2][2]; | |
| 789 | int i, j; | ||
| 790 | float det; | ||
| 791 | |||
| 792 | ✗ | out[0][0] = (m11 * m22 - m21 * m12); | |
| 793 | ✗ | out[0][1] = -(m01 * m22 - m21 * m02); | |
| 794 | ✗ | out[0][2] = (m01 * m12 - m11 * m02); | |
| 795 | ✗ | out[1][0] = -(m10 * m22 - m20 * m12); | |
| 796 | ✗ | out[1][1] = (m00 * m22 - m20 * m02); | |
| 797 | ✗ | out[1][2] = -(m00 * m12 - m10 * m02); | |
| 798 | ✗ | out[2][0] = (m10 * m21 - m20 * m11); | |
| 799 | ✗ | out[2][1] = -(m00 * m21 - m20 * m01); | |
| 800 | ✗ | out[2][2] = (m00 * m11 - m10 * m01); | |
| 801 | |||
| 802 | ✗ | det = m00 * out[0][0] + m10 * out[0][1] + m20 * out[0][2]; | |
| 803 | ✗ | det = 1.0 / det; | |
| 804 | |||
| 805 | ✗ | for (i = 0; i < 3; i++) { | |
| 806 | ✗ | for (j = 0; j < 3; j++) | |
| 807 | ✗ | out[i][j] *= det; | |
| 808 | } | ||
| 809 | ✗ | } | |
| 810 | |||
| 811 | ✗ | static void get_rgb2xyz_matrix(struct ColorSystem system, float m[3][3]) | |
| 812 | { | ||
| 813 | float S[3], X[4], Z[4]; | ||
| 814 | int i; | ||
| 815 | |||
| 816 | ✗ | X[0] = system.xRed / system.yRed; | |
| 817 | ✗ | X[1] = system.xGreen / system.yGreen; | |
| 818 | ✗ | X[2] = system.xBlue / system.yBlue; | |
| 819 | ✗ | X[3] = system.xWhite / system.yWhite; | |
| 820 | |||
| 821 | ✗ | Z[0] = (1 - system.xRed - system.yRed) / system.yRed; | |
| 822 | ✗ | Z[1] = (1 - system.xGreen - system.yGreen) / system.yGreen; | |
| 823 | ✗ | Z[2] = (1 - system.xBlue - system.yBlue) / system.yBlue; | |
| 824 | ✗ | Z[3] = (1 - system.xWhite - system.yWhite) / system.yWhite; | |
| 825 | |||
| 826 | ✗ | for (i = 0; i < 3; i++) { | |
| 827 | ✗ | m[0][i] = X[i]; | |
| 828 | ✗ | m[1][i] = 1; | |
| 829 | ✗ | m[2][i] = Z[i]; | |
| 830 | } | ||
| 831 | |||
| 832 | ✗ | invert_matrix3x3(m, m); | |
| 833 | |||
| 834 | ✗ | for (i = 0; i < 3; i++) | |
| 835 | ✗ | S[i] = m[i][0] * X[3] + m[i][1] * 1 + m[i][2] * Z[3]; | |
| 836 | |||
| 837 | ✗ | for (i = 0; i < 3; i++) { | |
| 838 | ✗ | m[0][i] = S[i] * X[i]; | |
| 839 | ✗ | m[1][i] = S[i] * 1; | |
| 840 | ✗ | m[2][i] = S[i] * Z[i]; | |
| 841 | } | ||
| 842 | ✗ | } | |
| 843 | |||
| 844 | static void | ||
| 845 | ✗ | rgb_to_xy(float rc, | |
| 846 | float gc, | ||
| 847 | float bc, | ||
| 848 | float * const x, | ||
| 849 | float * const y, | ||
| 850 | float * const z, | ||
| 851 | const float m[3][3]) | ||
| 852 | { | ||
| 853 | float scale; | ||
| 854 | |||
| 855 | ✗ | *x = m[0][0] * rc + m[0][1] * gc + m[0][2] * bc; | |
| 856 | ✗ | *y = m[1][0] * rc + m[1][1] * gc + m[1][2] * bc; | |
| 857 | ✗ | *z = m[2][0] * rc + m[2][1] * gc + m[2][2] * bc; | |
| 858 | |||
| 859 | ✗ | scale = *x + *y + *z; | |
| 860 | ✗ | scale = 1.f / scale; | |
| 861 | ✗ | *x = *x * scale; | |
| 862 | ✗ | *y = *y * scale; | |
| 863 | ✗ | } | |
| 864 | |||
| 865 | static int | ||
| 866 | ✗ | constrain_rgb(float * const r, | |
| 867 | float * const g, | ||
| 868 | float * const b) | ||
| 869 | { | ||
| 870 | /*---------------------------------------------------------------------------- | ||
| 871 | If the requested RGB shade contains a negative weight for one of | ||
| 872 | the primaries, it lies outside the color gamut accessible from | ||
| 873 | the given triple of primaries. Desaturate it by adding white, | ||
| 874 | equal quantities of R, G, and B, enough to make RGB all positive. | ||
| 875 | -----------------------------------------------------------------------------*/ | ||
| 876 | float w; | ||
| 877 | |||
| 878 | /* Amount of white needed is w = - min(0, *r, *g, *b) */ | ||
| 879 | ✗ | w = (0 < *r) ? 0 : *r; | |
| 880 | ✗ | w = (w < *g) ? w : *g; | |
| 881 | ✗ | w = (w < *b) ? w : *b; | |
| 882 | ✗ | w = - w; | |
| 883 | |||
| 884 | /* Add just enough white to make r, g, b all positive. */ | ||
| 885 | ✗ | if (w > 0) { | |
| 886 | ✗ | *r += w; *g += w; *b += w; | |
| 887 | |||
| 888 | ✗ | return 1; /* Color modified to fit RGB gamut */ | |
| 889 | } | ||
| 890 | |||
| 891 | ✗ | return 0; /* Color within RGB gamut */ | |
| 892 | } | ||
| 893 | |||
| 894 | static void | ||
| 895 | ✗ | gamma_correct(const struct ColorSystem * const cs, | |
| 896 | float * const c) | ||
| 897 | { | ||
| 898 | /*---------------------------------------------------------------------------- | ||
| 899 | Transform linear RGB values to nonlinear RGB values. | ||
| 900 | |||
| 901 | Rec. 709 is ITU-R Recommendation BT. 709 (1990) | ||
| 902 | ``Basic Parameter Values for the HDTV Standard for the Studio and for | ||
| 903 | International Programme Exchange'', formerly CCIR Rec. 709. | ||
| 904 | |||
| 905 | For details see | ||
| 906 | http://www.inforamp.net/~poynton/ColorFAQ.html | ||
| 907 | http://www.inforamp.net/~poynton/GammaFAQ.html | ||
| 908 | -----------------------------------------------------------------------------*/ | ||
| 909 | float gamma; | ||
| 910 | float cc; | ||
| 911 | |||
| 912 | ✗ | gamma = cs->gamma; | |
| 913 | |||
| 914 | ✗ | if (gamma == 0.) { | |
| 915 | /* Rec. 709 gamma correction. */ | ||
| 916 | ✗ | cc = 0.018; | |
| 917 | ✗ | if (*c < cc) { | |
| 918 | ✗ | *c *= (1.099 * pow(cc, 0.45) - 0.099) / cc; | |
| 919 | } else { | ||
| 920 | ✗ | *c = 1.099 * pow(*c, 0.45) - 0.099; | |
| 921 | } | ||
| 922 | } else { | ||
| 923 | /* Nonlinear color = (Linear color)^(1/gamma) */ | ||
| 924 | ✗ | *c = pow(*c, 1./gamma); | |
| 925 | } | ||
| 926 | ✗ | } | |
| 927 | |||
| 928 | |||
| 929 | |||
| 930 | static void | ||
| 931 | ✗ | gamma_correct_rgb(const struct ColorSystem * const cs, | |
| 932 | float * const r, | ||
| 933 | float * const g, | ||
| 934 | float * const b) | ||
| 935 | { | ||
| 936 | ✗ | gamma_correct(cs, r); | |
| 937 | ✗ | gamma_correct(cs, g); | |
| 938 | ✗ | gamma_correct(cs, b); | |
| 939 | ✗ | } | |
| 940 | |||
| 941 | /* Sz(X) is the displacement in pixels of a displacement of X normalized | ||
| 942 | distance units. (A normalized distance unit is 1/512 of the smaller | ||
| 943 | dimension of the canvas) | ||
| 944 | */ | ||
| 945 | #define Sz(x) (((x) * (int)FFMIN(w, h)) / 512) | ||
| 946 | |||
| 947 | static void | ||
| 948 | ✗ | monochrome_color_location(float waveLength, int w, int h, | |
| 949 | int cie, int *xP, int *yP) | ||
| 950 | { | ||
| 951 | ✗ | const int ix = waveLength - 360; | |
| 952 | ✗ | const float pX = spectral_chromaticity[ix][0]; | |
| 953 | ✗ | const float pY = spectral_chromaticity[ix][1]; | |
| 954 | ✗ | const float pZ = spectral_chromaticity[ix][2]; | |
| 955 | ✗ | const float px = pX / (pX + pY + pZ); | |
| 956 | ✗ | const float py = pY / (pX + pY + pZ); | |
| 957 | |||
| 958 | ✗ | if (cie == LUV) { | |
| 959 | float up, vp; | ||
| 960 | |||
| 961 | ✗ | xy_to_upvp(px, py, &up, &vp); | |
| 962 | ✗ | *xP = up * (w - 1); | |
| 963 | ✗ | *yP = (h - 1) - vp * (h - 1); | |
| 964 | ✗ | } else if (cie == UCS) { | |
| 965 | float u, v; | ||
| 966 | |||
| 967 | ✗ | xy_to_uv(px, py, &u, &v); | |
| 968 | ✗ | *xP = u * (w - 1); | |
| 969 | ✗ | *yP = (h - 1) - v * (h - 1); | |
| 970 | ✗ | } else if (cie == XYY) { | |
| 971 | ✗ | *xP = px * (w - 1); | |
| 972 | ✗ | *yP = (h - 1) - py * (h - 1); | |
| 973 | } else { | ||
| 974 | ✗ | av_assert0(0); | |
| 975 | } | ||
| 976 | ✗ | } | |
| 977 | |||
| 978 | static void | ||
| 979 | ✗ | find_tongue(uint16_t* const pixels, | |
| 980 | int const w, | ||
| 981 | int const linesize, | ||
| 982 | int const row, | ||
| 983 | int * const presentP, | ||
| 984 | int * const leftEdgeP, | ||
| 985 | int * const rightEdgeP) | ||
| 986 | { | ||
| 987 | int i; | ||
| 988 | |||
| 989 | ✗ | for (i = 0; i < w && pixels[row * linesize + i * 4 + 0] == 0; i++) | |
| 990 | ; | ||
| 991 | |||
| 992 | ✗ | if (i >= w) { | |
| 993 | ✗ | *presentP = 0; | |
| 994 | } else { | ||
| 995 | int j; | ||
| 996 | ✗ | int const leftEdge = i; | |
| 997 | |||
| 998 | ✗ | *presentP = 1; | |
| 999 | |||
| 1000 | ✗ | for (j = w - 1; j >= leftEdge && pixels[row * linesize + j * 4 + 0] == 0; j--) | |
| 1001 | ; | ||
| 1002 | |||
| 1003 | ✗ | *rightEdgeP = j; | |
| 1004 | ✗ | *leftEdgeP = leftEdge; | |
| 1005 | } | ||
| 1006 | ✗ | } | |
| 1007 | |||
| 1008 | ✗ | static void draw_line(uint16_t *const pixels, int linesize, | |
| 1009 | int x0, int y0, int x1, int y1, | ||
| 1010 | int w, int h, | ||
| 1011 | const uint16_t *const rgbcolor) | ||
| 1012 | { | ||
| 1013 | ✗ | int sx = x0 < x1 ? 1 : -1, sy = y0 < y1 ? 1 : -1, x2; | |
| 1014 | ✗ | int dx = FFABS(x1-x0), dy = FFABS(y1-y0), err = dx * dx + dy * dy; | |
| 1015 | ✗ | int e2 = err == 0 ? 1 : 0xffffff / (dx + dy); | |
| 1016 | |||
| 1017 | ✗ | dx *= e2; | |
| 1018 | ✗ | dy *= e2; | |
| 1019 | ✗ | err = dx - dy; | |
| 1020 | |||
| 1021 | for (;;) { | ||
| 1022 | ✗ | pixels[y0 * linesize + x0 * 4 + 0] = rgbcolor[0]-(FFABS(err - dx + dy) >> 8); | |
| 1023 | ✗ | pixels[y0 * linesize + x0 * 4 + 1] = rgbcolor[1]-(FFABS(err - dx + dy) >> 8); | |
| 1024 | ✗ | pixels[y0 * linesize + x0 * 4 + 2] = rgbcolor[2]-(FFABS(err - dx + dy) >> 8); | |
| 1025 | ✗ | pixels[y0 * linesize + x0 * 4 + 3] = rgbcolor[3]-(FFABS(err - dx + dy) >> 8); | |
| 1026 | |||
| 1027 | ✗ | e2 = err; | |
| 1028 | ✗ | x2 = x0; | |
| 1029 | ✗ | if (2 * e2 >= -dx) { | |
| 1030 | ✗ | if (x0 == x1) | |
| 1031 | ✗ | break; | |
| 1032 | ✗ | if (e2 + dy < 0xff0000) { | |
| 1033 | ✗ | pixels[(y0 + sy) * linesize + x0 * 4 + 0] = rgbcolor[0]-(FFABS(e2 + dy) >> 8); | |
| 1034 | ✗ | pixels[(y0 + sy) * linesize + x0 * 4 + 1] = rgbcolor[1]-(FFABS(e2 + dy) >> 8); | |
| 1035 | ✗ | pixels[(y0 + sy) * linesize + x0 * 4 + 2] = rgbcolor[2]-(FFABS(e2 + dy) >> 8); | |
| 1036 | ✗ | pixels[(y0 + sy) * linesize + x0 * 4 + 3] = rgbcolor[3]-(FFABS(e2 + dy) >> 8); | |
| 1037 | } | ||
| 1038 | ✗ | err -= dy; | |
| 1039 | ✗ | x0 += sx; | |
| 1040 | } | ||
| 1041 | |||
| 1042 | ✗ | if (2 * e2 <= dy) { | |
| 1043 | ✗ | if (y0 == y1) | |
| 1044 | ✗ | break; | |
| 1045 | ✗ | if (dx - e2 < 0xff0000) { | |
| 1046 | ✗ | pixels[y0 * linesize + (x2 + sx) * 4 + 0] = rgbcolor[0]-(FFABS(dx - e2) >> 8); | |
| 1047 | ✗ | pixels[y0 * linesize + (x2 + sx) * 4 + 1] = rgbcolor[1]-(FFABS(dx - e2) >> 8); | |
| 1048 | ✗ | pixels[y0 * linesize + (x2 + sx) * 4 + 2] = rgbcolor[2]-(FFABS(dx - e2) >> 8); | |
| 1049 | ✗ | pixels[y0 * linesize + (x2 + sx) * 4 + 3] = rgbcolor[3]-(FFABS(dx - e2) >> 8); | |
| 1050 | } | ||
| 1051 | ✗ | err += dx; | |
| 1052 | ✗ | y0 += sy; | |
| 1053 | } | ||
| 1054 | } | ||
| 1055 | ✗ | } | |
| 1056 | |||
| 1057 | ✗ | static void draw_rline(uint16_t *const pixels, int linesize, | |
| 1058 | int x0, int y0, int x1, int y1, | ||
| 1059 | int w, int h) | ||
| 1060 | { | ||
| 1061 | ✗ | int dx = FFABS(x1 - x0), sx = x0 < x1 ? 1 : -1; | |
| 1062 | ✗ | int dy = FFABS(y1 - y0), sy = y0 < y1 ? 1 : -1; | |
| 1063 | ✗ | int err = (dx > dy ? dx : -dy) / 2, e2; | |
| 1064 | |||
| 1065 | for (;;) { | ||
| 1066 | ✗ | pixels[y0 * linesize + x0 * 4 + 0] = 65535 - pixels[y0 * linesize + x0 * 4 + 0]; | |
| 1067 | ✗ | pixels[y0 * linesize + x0 * 4 + 1] = 65535 - pixels[y0 * linesize + x0 * 4 + 1]; | |
| 1068 | ✗ | pixels[y0 * linesize + x0 * 4 + 2] = 65535 - pixels[y0 * linesize + x0 * 4 + 2]; | |
| 1069 | ✗ | pixels[y0 * linesize + x0 * 4 + 3] = 65535; | |
| 1070 | |||
| 1071 | ✗ | if (x0 == x1 && y0 == y1) | |
| 1072 | ✗ | break; | |
| 1073 | |||
| 1074 | ✗ | e2 = err; | |
| 1075 | |||
| 1076 | ✗ | if (e2 >-dx) { | |
| 1077 | ✗ | err -= dy; | |
| 1078 | ✗ | x0 += sx; | |
| 1079 | } | ||
| 1080 | |||
| 1081 | ✗ | if (e2 < dy) { | |
| 1082 | ✗ | err += dx; | |
| 1083 | ✗ | y0 += sy; | |
| 1084 | } | ||
| 1085 | } | ||
| 1086 | ✗ | } | |
| 1087 | |||
| 1088 | static void | ||
| 1089 | ✗ | tongue_outline(uint16_t* const pixels, | |
| 1090 | int const linesize, | ||
| 1091 | int const w, | ||
| 1092 | int const h, | ||
| 1093 | uint16_t const maxval, | ||
| 1094 | int const cie) | ||
| 1095 | { | ||
| 1096 | ✗ | const uint16_t rgbcolor[4] = { maxval, maxval, maxval, maxval }; | |
| 1097 | int wavelength; | ||
| 1098 | int lx, ly; | ||
| 1099 | int fx, fy; | ||
| 1100 | |||
| 1101 | ✗ | for (wavelength = 360; wavelength <= 830; wavelength++) { | |
| 1102 | int icx, icy; | ||
| 1103 | |||
| 1104 | ✗ | monochrome_color_location(wavelength, w, h, cie, | |
| 1105 | &icx, &icy); | ||
| 1106 | |||
| 1107 | ✗ | if (wavelength > 360) | |
| 1108 | ✗ | draw_line(pixels, linesize, lx, ly, icx, icy, w, h, rgbcolor); | |
| 1109 | else { | ||
| 1110 | ✗ | fx = icx; | |
| 1111 | ✗ | fy = icy; | |
| 1112 | } | ||
| 1113 | ✗ | lx = icx; | |
| 1114 | ✗ | ly = icy; | |
| 1115 | } | ||
| 1116 | ✗ | draw_line(pixels, linesize, lx, ly, fx, fy, w, h, rgbcolor); | |
| 1117 | ✗ | } | |
| 1118 | |||
| 1119 | static void | ||
| 1120 | ✗ | fill_in_tongue(uint16_t* const pixels, | |
| 1121 | int const linesize, | ||
| 1122 | int const w, | ||
| 1123 | int const h, | ||
| 1124 | uint16_t const maxval, | ||
| 1125 | const struct ColorSystem * const cs, | ||
| 1126 | float const m[3][3], | ||
| 1127 | int const cie, | ||
| 1128 | int const correct_gamma, | ||
| 1129 | float const contrast) | ||
| 1130 | { | ||
| 1131 | int y; | ||
| 1132 | |||
| 1133 | /* Scan the image line by line and fill the tongue outline | ||
| 1134 | with the RGB values determined by the color system for the x-y | ||
| 1135 | coordinates within the tongue. | ||
| 1136 | */ | ||
| 1137 | |||
| 1138 | ✗ | for (y = 0; y < h; ++y) { | |
| 1139 | int present; /* There is some tongue on this line */ | ||
| 1140 | int leftEdge; /* x position of leftmost pixel in tongue on this line */ | ||
| 1141 | int rightEdge; /* same, but rightmost */ | ||
| 1142 | |||
| 1143 | ✗ | find_tongue(pixels, w, linesize, y, &present, &leftEdge, &rightEdge); | |
| 1144 | |||
| 1145 | ✗ | if (present) { | |
| 1146 | int x; | ||
| 1147 | |||
| 1148 | ✗ | for (x = leftEdge; x <= rightEdge; ++x) { | |
| 1149 | float cx, cy, cz, jr, jg, jb, jmax; | ||
| 1150 | ✗ | int r, g, b, mx = maxval; | |
| 1151 | |||
| 1152 | ✗ | if (cie == LUV) { | |
| 1153 | float up, vp; | ||
| 1154 | ✗ | up = ((float) x) / (w - 1); | |
| 1155 | ✗ | vp = 1.0 - ((float) y) / (h - 1); | |
| 1156 | ✗ | upvp_to_xy(up, vp, &cx, &cy); | |
| 1157 | ✗ | cz = 1.0 - (cx + cy); | |
| 1158 | ✗ | } else if (cie == UCS) { | |
| 1159 | float u, v; | ||
| 1160 | ✗ | u = ((float) x) / (w - 1); | |
| 1161 | ✗ | v = 1.0 - ((float) y) / (h - 1); | |
| 1162 | ✗ | uv_to_xy(u, v, &cx, &cy); | |
| 1163 | ✗ | cz = 1.0 - (cx + cy); | |
| 1164 | ✗ | } else if (cie == XYY) { | |
| 1165 | ✗ | cx = ((float) x) / (w - 1); | |
| 1166 | ✗ | cy = 1.0 - ((float) y) / (h - 1); | |
| 1167 | ✗ | cz = 1.0 - (cx + cy); | |
| 1168 | } else { | ||
| 1169 | ✗ | av_assert0(0); | |
| 1170 | } | ||
| 1171 | |||
| 1172 | ✗ | xyz_to_rgb(m, cx, cy, cz, &jr, &jg, &jb); | |
| 1173 | |||
| 1174 | /* Check whether the requested color is within the | ||
| 1175 | gamut achievable with the given color system. If | ||
| 1176 | not, draw it in a reduced intensity, interpolated | ||
| 1177 | by desaturation to the closest within-gamut color. */ | ||
| 1178 | |||
| 1179 | ✗ | if (constrain_rgb(&jr, &jg, &jb)) | |
| 1180 | ✗ | mx *= contrast; | |
| 1181 | |||
| 1182 | ✗ | jmax = FFMAX3(jr, jg, jb); | |
| 1183 | ✗ | if (jmax > 0) { | |
| 1184 | ✗ | jr = jr / jmax; | |
| 1185 | ✗ | jg = jg / jmax; | |
| 1186 | ✗ | jb = jb / jmax; | |
| 1187 | } | ||
| 1188 | /* gamma correct from linear rgb to nonlinear rgb. */ | ||
| 1189 | ✗ | if (correct_gamma) | |
| 1190 | ✗ | gamma_correct_rgb(cs, &jr, &jg, &jb); | |
| 1191 | ✗ | r = mx * jr; | |
| 1192 | ✗ | g = mx * jg; | |
| 1193 | ✗ | b = mx * jb; | |
| 1194 | ✗ | pixels[y * linesize + x * 4 + 0] = r; | |
| 1195 | ✗ | pixels[y * linesize + x * 4 + 1] = g; | |
| 1196 | ✗ | pixels[y * linesize + x * 4 + 2] = b; | |
| 1197 | ✗ | pixels[y * linesize + x * 4 + 3] = 65535; | |
| 1198 | } | ||
| 1199 | } | ||
| 1200 | } | ||
| 1201 | ✗ | } | |
| 1202 | |||
| 1203 | static void | ||
| 1204 | ✗ | plot_white_point(uint16_t* pixels, | |
| 1205 | int const linesize, | ||
| 1206 | int const w, | ||
| 1207 | int const h, | ||
| 1208 | int const maxval, | ||
| 1209 | int const color_system, | ||
| 1210 | int const cie) | ||
| 1211 | { | ||
| 1212 | ✗ | const struct ColorSystem *cs = &color_systems[color_system]; | |
| 1213 | int wx, wy; | ||
| 1214 | |||
| 1215 | ✗ | if (cie == LUV) { | |
| 1216 | float wup, wvp; | ||
| 1217 | ✗ | xy_to_upvp(cs->xWhite, cs->yWhite, &wup, &wvp); | |
| 1218 | ✗ | wx = (w - 1) * wup; | |
| 1219 | ✗ | wy = (h - 1) - ((int) ((h - 1) * wvp)); | |
| 1220 | ✗ | } else if (cie == UCS) { | |
| 1221 | float wu, wv; | ||
| 1222 | ✗ | xy_to_uv(cs->xWhite, cs->yWhite, &wu, &wv); | |
| 1223 | ✗ | wx = (w - 1) * wu; | |
| 1224 | ✗ | wy = (h - 1) - ((int) ((h - 1) * wv)); | |
| 1225 | ✗ | } else if (cie == XYY) { | |
| 1226 | ✗ | wx = (w - 1) * cs->xWhite; | |
| 1227 | ✗ | wy = (h - 1) - ((int) ((h - 1) * cs->yWhite)); | |
| 1228 | } else { | ||
| 1229 | ✗ | av_assert0(0); | |
| 1230 | } | ||
| 1231 | |||
| 1232 | ✗ | draw_rline(pixels, linesize, | |
| 1233 | ✗ | wx + Sz(3), wy, wx + Sz(10), wy, | |
| 1234 | w, h); | ||
| 1235 | ✗ | draw_rline(pixels, linesize, | |
| 1236 | ✗ | wx - Sz(3), wy, wx - Sz(10), wy, | |
| 1237 | w, h); | ||
| 1238 | ✗ | draw_rline(pixels, linesize, | |
| 1239 | ✗ | wx, wy + Sz(3), wx, wy + Sz(10), | |
| 1240 | w, h); | ||
| 1241 | ✗ | draw_rline(pixels, linesize, | |
| 1242 | ✗ | wx, wy - Sz(3), wx, wy - Sz(10), | |
| 1243 | w, h); | ||
| 1244 | ✗ | } | |
| 1245 | |||
| 1246 | ✗ | static int draw_background(AVFilterContext *ctx) | |
| 1247 | { | ||
| 1248 | ✗ | CiescopeContext *s = ctx->priv; | |
| 1249 | ✗ | const struct ColorSystem *cs = &color_systems[s->color_system]; | |
| 1250 | ✗ | AVFilterLink *outlink = ctx->outputs[0]; | |
| 1251 | ✗ | int w = s->size; | |
| 1252 | ✗ | int h = s->size; | |
| 1253 | uint16_t *pixels; | ||
| 1254 | |||
| 1255 | ✗ | if ((s->f = ff_get_video_buffer(outlink, outlink->w, outlink->h)) == NULL) | |
| 1256 | ✗ | return AVERROR(ENOMEM); | |
| 1257 | ✗ | pixels = (uint16_t *)s->f->data[0]; | |
| 1258 | |||
| 1259 | ✗ | tongue_outline(pixels, s->f->linesize[0] / 2, w, h, 65535, s->cie); | |
| 1260 | |||
| 1261 | ✗ | if (s->fill) | |
| 1262 | ✗ | fill_in_tongue(pixels, s->f->linesize[0] / 2, w, h, 65535, cs, (const float (*)[3])s->i, s->cie, | |
| 1263 | s->correct_gamma, s->contrast); | ||
| 1264 | |||
| 1265 | ✗ | return 0; | |
| 1266 | } | ||
| 1267 | |||
| 1268 | ✗ | static void filter_rgb48(AVFilterContext *ctx, const uint8_t *ptr, | |
| 1269 | ptrdiff_t linesize, | ||
| 1270 | float *cx, float *cy, int x, int y) | ||
| 1271 | { | ||
| 1272 | ✗ | CiescopeContext *s = ctx->priv; | |
| 1273 | ✗ | const float scale = 1.f / 65535.f; | |
| 1274 | ✗ | const uint16_t *src = (const uint16_t*)(ptr + linesize * y + x * 6); | |
| 1275 | ✗ | float r = (src[0] + 0.01f) * scale; | |
| 1276 | ✗ | float g = (src[1] + 0.01f) * scale; | |
| 1277 | ✗ | float b = (src[2] + 0.01f) * scale; | |
| 1278 | float cz; | ||
| 1279 | |||
| 1280 | ✗ | rgb_to_xy(r, g, b, cx, cy, &cz, (const float (*)[3])s->m); | |
| 1281 | ✗ | } | |
| 1282 | |||
| 1283 | ✗ | static void filter_rgba64(AVFilterContext *ctx, const uint8_t *ptr, | |
| 1284 | ptrdiff_t linesize, | ||
| 1285 | float *cx, float *cy, int x, int y) | ||
| 1286 | { | ||
| 1287 | ✗ | CiescopeContext *s = ctx->priv; | |
| 1288 | ✗ | const float scale = 1.f / 65535.f; | |
| 1289 | ✗ | const uint16_t *src = (const uint16_t*)(ptr + linesize * y + x * 8); | |
| 1290 | ✗ | float r = (src[0] + 0.01f) * scale; | |
| 1291 | ✗ | float g = (src[1] + 0.01f) * scale; | |
| 1292 | ✗ | float b = (src[2] + 0.01f) * scale; | |
| 1293 | float cz; | ||
| 1294 | |||
| 1295 | ✗ | rgb_to_xy(r, g, b, cx, cy, &cz, (const float (*)[3])s->m); | |
| 1296 | ✗ | } | |
| 1297 | |||
| 1298 | ✗ | static void filter_rgb24(AVFilterContext *ctx, const uint8_t *ptr, | |
| 1299 | ptrdiff_t linesize, | ||
| 1300 | float *cx, float *cy, int x, int y) | ||
| 1301 | { | ||
| 1302 | ✗ | CiescopeContext *s = ctx->priv; | |
| 1303 | ✗ | const float scale = 1.f / 255.f; | |
| 1304 | ✗ | const uint8_t *src = ptr + linesize * y + x * 3; | |
| 1305 | ✗ | float r = (src[0] + 0.01f) * scale; | |
| 1306 | ✗ | float g = (src[1] + 0.01f) * scale; | |
| 1307 | ✗ | float b = (src[2] + 0.01f) * scale; | |
| 1308 | float cz; | ||
| 1309 | |||
| 1310 | ✗ | rgb_to_xy(r, g, b, cx, cy, &cz, (const float (*)[3])s->m); | |
| 1311 | ✗ | } | |
| 1312 | |||
| 1313 | ✗ | static void filter_rgba(AVFilterContext *ctx, const uint8_t *ptr, | |
| 1314 | ptrdiff_t linesize, | ||
| 1315 | float *cx, float *cy, int x, int y) | ||
| 1316 | { | ||
| 1317 | ✗ | CiescopeContext *s = ctx->priv; | |
| 1318 | ✗ | const float scale = 1.f / 255.f; | |
| 1319 | ✗ | const uint8_t *src = ptr + linesize * y + x * 4; | |
| 1320 | ✗ | float r = (src[0] + 0.01f) * scale; | |
| 1321 | ✗ | float g = (src[1] + 0.01f) * scale; | |
| 1322 | ✗ | float b = (src[2] + 0.01f) * scale; | |
| 1323 | float cz; | ||
| 1324 | |||
| 1325 | ✗ | rgb_to_xy(r, g, b, cx, cy, &cz, (const float (*)[3])s->m); | |
| 1326 | ✗ | } | |
| 1327 | |||
| 1328 | ✗ | static void filter_xyz(AVFilterContext *ctx, const uint8_t *ptr, | |
| 1329 | ptrdiff_t linesize, | ||
| 1330 | float *cx, float *cy, int x, int y) | ||
| 1331 | { | ||
| 1332 | ✗ | CiescopeContext *s = ctx->priv; | |
| 1333 | ✗ | const uint16_t* src = (uint16_t *)(ptr + linesize * y + x * 6); | |
| 1334 | ✗ | float lx = s->log2lin[src[0]]; | |
| 1335 | ✗ | float ly = s->log2lin[src[1]]; | |
| 1336 | ✗ | float lz = s->log2lin[src[2]]; | |
| 1337 | ✗ | float sum = lx + ly + lz; | |
| 1338 | |||
| 1339 | ✗ | if (sum == 0) | |
| 1340 | ✗ | sum = 1; | |
| 1341 | ✗ | *cx = lx / sum; | |
| 1342 | ✗ | *cy = ly / sum; | |
| 1343 | ✗ | } | |
| 1344 | |||
| 1345 | ✗ | static void plot_gamuts(uint16_t *pixels, int linesize, int w, int h, | |
| 1346 | int cie, int gamuts) | ||
| 1347 | { | ||
| 1348 | int i; | ||
| 1349 | |||
| 1350 | ✗ | for (i = 0; i < NB_CS; i++) { | |
| 1351 | ✗ | const struct ColorSystem *cs = &color_systems[i]; | |
| 1352 | int rx, ry, gx, gy, bx, by; | ||
| 1353 | |||
| 1354 | ✗ | if (!((1 << i) & gamuts)) | |
| 1355 | ✗ | continue; | |
| 1356 | ✗ | if (cie == LUV) { | |
| 1357 | float wup, wvp; | ||
| 1358 | ✗ | xy_to_upvp(cs->xRed, cs->yRed, &wup, &wvp); | |
| 1359 | ✗ | rx = (w - 1) * wup; | |
| 1360 | ✗ | ry = (h - 1) - ((int) ((h - 1) * wvp)); | |
| 1361 | ✗ | xy_to_upvp(cs->xGreen, cs->yGreen, &wup, &wvp); | |
| 1362 | ✗ | gx = (w - 1) * wup; | |
| 1363 | ✗ | gy = (h - 1) - ((int) ((h - 1) * wvp)); | |
| 1364 | ✗ | xy_to_upvp(cs->xBlue, cs->yBlue, &wup, &wvp); | |
| 1365 | ✗ | bx = (w - 1) * wup; | |
| 1366 | ✗ | by = (h - 1) - ((int) ((h - 1) * wvp)); | |
| 1367 | ✗ | } else if (cie == UCS) { | |
| 1368 | float wu, wv; | ||
| 1369 | ✗ | xy_to_uv(cs->xRed, cs->yRed, &wu, &wv); | |
| 1370 | ✗ | rx = (w - 1) * wu; | |
| 1371 | ✗ | ry = (h - 1) - ((int) ((h - 1) * wv)); | |
| 1372 | ✗ | xy_to_uv(cs->xGreen, cs->yGreen, &wu, &wv); | |
| 1373 | ✗ | gx = (w - 1) * wu; | |
| 1374 | ✗ | gy = (h - 1) - ((int) ((h - 1) * wv)); | |
| 1375 | ✗ | xy_to_uv(cs->xBlue, cs->yBlue, &wu, &wv); | |
| 1376 | ✗ | bx = (w - 1) * wu; | |
| 1377 | ✗ | by = (h - 1) - ((int) ((h - 1) * wv)); | |
| 1378 | ✗ | } else if (cie == XYY) { | |
| 1379 | ✗ | rx = (w - 1) * cs->xRed; | |
| 1380 | ✗ | ry = (h - 1) - ((int) ((h - 1) * cs->yRed)); | |
| 1381 | ✗ | gx = (w - 1) * cs->xGreen; | |
| 1382 | ✗ | gy = (h - 1) - ((int) ((h - 1) * cs->yGreen)); | |
| 1383 | ✗ | bx = (w - 1) * cs->xBlue; | |
| 1384 | ✗ | by = (h - 1) - ((int) ((h - 1) * cs->yBlue)); | |
| 1385 | } else { | ||
| 1386 | ✗ | av_assert0(0); | |
| 1387 | } | ||
| 1388 | |||
| 1389 | ✗ | draw_rline(pixels, linesize, rx, ry, gx, gy, w, h); | |
| 1390 | ✗ | draw_rline(pixels, linesize, gx, gy, bx, by, w, h); | |
| 1391 | ✗ | draw_rline(pixels, linesize, bx, by, rx, ry, w, h); | |
| 1392 | } | ||
| 1393 | ✗ | } | |
| 1394 | |||
| 1395 | ✗ | static int filter_frame(AVFilterLink *inlink, AVFrame *in) | |
| 1396 | { | ||
| 1397 | ✗ | AVFilterContext *ctx = inlink->dst; | |
| 1398 | ✗ | CiescopeContext *s = ctx->priv; | |
| 1399 | ✗ | AVFilterLink *outlink = ctx->outputs[0]; | |
| 1400 | ✗ | int i = s->intensity * 65535; | |
| 1401 | ✗ | int w = outlink->w; | |
| 1402 | ✗ | int h = outlink->h; | |
| 1403 | AVFrame *out; | ||
| 1404 | int ret, x, y; | ||
| 1405 | |||
| 1406 | ✗ | out = ff_get_video_buffer(outlink, outlink->w, outlink->h); | |
| 1407 | ✗ | if (!out) { | |
| 1408 | ✗ | av_frame_free(&in); | |
| 1409 | ✗ | return AVERROR(ENOMEM); | |
| 1410 | } | ||
| 1411 | ✗ | out->pts = in->pts; | |
| 1412 | ✗ | out->duration = in->duration; | |
| 1413 | |||
| 1414 | ✗ | if (!s->background) { | |
| 1415 | ✗ | ret = draw_background(ctx); | |
| 1416 | ✗ | if (ret < 0) { | |
| 1417 | ✗ | av_frame_free(&out); | |
| 1418 | ✗ | return ret; | |
| 1419 | } | ||
| 1420 | ✗ | s->background = 1; | |
| 1421 | } | ||
| 1422 | ✗ | for (y = 0; y < outlink->h; y++) { | |
| 1423 | ✗ | memset(out->data[0] + y * out->linesize[0], 0, outlink->w * 8); | |
| 1424 | } | ||
| 1425 | |||
| 1426 | ✗ | for (y = 0; y < in->height; y++) { | |
| 1427 | ✗ | const uint8_t *src = in->data[0]; | |
| 1428 | ✗ | const ptrdiff_t src_linesize = in->linesize[0]; | |
| 1429 | ✗ | uint16_t *dst = (uint16_t *)out->data[0]; | |
| 1430 | ✗ | const ptrdiff_t linesize = out->linesize[0] / 2; | |
| 1431 | ✗ | const int w_1 = w - 1; | |
| 1432 | ✗ | const int h_1 = h - 1; | |
| 1433 | |||
| 1434 | ✗ | for (x = 0; x < in->width; x++) { | |
| 1435 | float cx, cy; | ||
| 1436 | int wx, wy, pos; | ||
| 1437 | int r, g, b; | ||
| 1438 | |||
| 1439 | ✗ | s->filter(ctx, src, src_linesize, &cx, &cy, x, y); | |
| 1440 | |||
| 1441 | ✗ | if (s->cie == LUV) { | |
| 1442 | float up, vp; | ||
| 1443 | ✗ | xy_to_upvp(cx, cy, &up, &vp); | |
| 1444 | ✗ | cx = up; | |
| 1445 | ✗ | cy = vp; | |
| 1446 | ✗ | } else if (s->cie == UCS) { | |
| 1447 | float u, v; | ||
| 1448 | ✗ | xy_to_uv(cx, cy, &u, &v); | |
| 1449 | ✗ | cx = u; | |
| 1450 | ✗ | cy = v; | |
| 1451 | } | ||
| 1452 | |||
| 1453 | ✗ | wx = w_1 * cx; | |
| 1454 | ✗ | wy = h_1 - h_1 * cy; | |
| 1455 | |||
| 1456 | ✗ | if (wx < 0 || wx >= w || | |
| 1457 | ✗ | wy < 0 || wy >= h) | |
| 1458 | ✗ | continue; | |
| 1459 | |||
| 1460 | ✗ | pos = wy * linesize + wx * 4; | |
| 1461 | ✗ | r = dst[pos + 0] + i; | |
| 1462 | ✗ | g = dst[pos + 1] + i; | |
| 1463 | ✗ | b = dst[pos + 2] + i; | |
| 1464 | |||
| 1465 | ✗ | dst[pos + 0] = FFMIN(r, 65535); | |
| 1466 | ✗ | dst[pos + 1] = FFMIN(g, 65535); | |
| 1467 | ✗ | dst[pos + 2] = FFMIN(b, 65535); | |
| 1468 | ✗ | dst[pos + 3] = 65535; | |
| 1469 | } | ||
| 1470 | } | ||
| 1471 | |||
| 1472 | ✗ | for (y = 0; y < outlink->h; y++) { | |
| 1473 | ✗ | uint16_t *dst = (uint16_t *)(out->data[0] + y * out->linesize[0]); | |
| 1474 | ✗ | const uint16_t *src = (const uint16_t *)(s->f->data[0] + y * s->f->linesize[0]); | |
| 1475 | ✗ | for (x = 0; x < outlink->w; x++) { | |
| 1476 | ✗ | const int xx = x * 4; | |
| 1477 | ✗ | if (dst[xx + 3] == 0) { | |
| 1478 | ✗ | dst[xx + 0] = src[xx + 0]; | |
| 1479 | ✗ | dst[xx + 1] = src[xx + 1]; | |
| 1480 | ✗ | dst[xx + 2] = src[xx + 2]; | |
| 1481 | ✗ | dst[xx + 3] = src[xx + 3]; | |
| 1482 | } | ||
| 1483 | } | ||
| 1484 | } | ||
| 1485 | |||
| 1486 | ✗ | if (s->show_white) | |
| 1487 | ✗ | plot_white_point((uint16_t *)out->data[0], out->linesize[0] / 2, | |
| 1488 | outlink->w, outlink->h, 65535, | ||
| 1489 | s->color_system, s->cie); | ||
| 1490 | |||
| 1491 | ✗ | plot_gamuts((uint16_t *)out->data[0], out->linesize[0] / 2, | |
| 1492 | outlink->w, outlink->h, | ||
| 1493 | ✗ | s->cie, s->gamuts); | |
| 1494 | |||
| 1495 | ✗ | av_frame_free(&in); | |
| 1496 | ✗ | return ff_filter_frame(outlink, out); | |
| 1497 | } | ||
| 1498 | |||
| 1499 | ✗ | static void av_cold uninit(AVFilterContext *ctx) | |
| 1500 | { | ||
| 1501 | ✗ | CiescopeContext *s = ctx->priv; | |
| 1502 | |||
| 1503 | ✗ | av_frame_free(&s->f); | |
| 1504 | ✗ | } | |
| 1505 | |||
| 1506 | ✗ | static int config_input(AVFilterLink *inlink) | |
| 1507 | { | ||
| 1508 | ✗ | CiescopeContext *s = inlink->dst->priv; | |
| 1509 | int i; | ||
| 1510 | |||
| 1511 | ✗ | get_rgb2xyz_matrix(color_systems[s->color_system], s->m); | |
| 1512 | ✗ | invert_matrix3x3(s->m, s->i); | |
| 1513 | |||
| 1514 | ✗ | switch (inlink->format) { | |
| 1515 | ✗ | case AV_PIX_FMT_RGB24: | |
| 1516 | ✗ | s->filter = filter_rgb24; | |
| 1517 | ✗ | break; | |
| 1518 | ✗ | case AV_PIX_FMT_RGBA: | |
| 1519 | ✗ | s->filter = filter_rgba; | |
| 1520 | ✗ | break; | |
| 1521 | ✗ | case AV_PIX_FMT_RGB48: | |
| 1522 | ✗ | s->filter = filter_rgb48; | |
| 1523 | ✗ | break; | |
| 1524 | ✗ | case AV_PIX_FMT_RGBA64: | |
| 1525 | ✗ | s->filter = filter_rgba64; | |
| 1526 | ✗ | break; | |
| 1527 | ✗ | case AV_PIX_FMT_XYZ12: | |
| 1528 | ✗ | s->filter = filter_xyz; | |
| 1529 | ✗ | for (i = 0; i < 65536; i++) | |
| 1530 | ✗ | s->log2lin[i] = pow(i / 65535., s->igamma) * 65535.; | |
| 1531 | ✗ | break; | |
| 1532 | ✗ | default: | |
| 1533 | ✗ | av_assert0(0); | |
| 1534 | } | ||
| 1535 | |||
| 1536 | ✗ | return 0; | |
| 1537 | } | ||
| 1538 | |||
| 1539 | static const AVFilterPad inputs[] = { | ||
| 1540 | { | ||
| 1541 | .name = "default", | ||
| 1542 | .type = AVMEDIA_TYPE_VIDEO, | ||
| 1543 | .filter_frame = filter_frame, | ||
| 1544 | .config_props = config_input, | ||
| 1545 | }, | ||
| 1546 | }; | ||
| 1547 | |||
| 1548 | static const AVFilterPad outputs[] = { | ||
| 1549 | { | ||
| 1550 | .name = "default", | ||
| 1551 | .type = AVMEDIA_TYPE_VIDEO, | ||
| 1552 | .config_props = config_output, | ||
| 1553 | }, | ||
| 1554 | }; | ||
| 1555 | |||
| 1556 | const FFFilter ff_vf_ciescope = { | ||
| 1557 | .p.name = "ciescope", | ||
| 1558 | .p.description = NULL_IF_CONFIG_SMALL("Video CIE scope."), | ||
| 1559 | .p.priv_class = &ciescope_class, | ||
| 1560 | .priv_size = sizeof(CiescopeContext), | ||
| 1561 | .uninit = uninit, | ||
| 1562 | FILTER_INPUTS(inputs), | ||
| 1563 | FILTER_OUTPUTS(outputs), | ||
| 1564 | FILTER_QUERY_FUNC2(query_formats), | ||
| 1565 | }; | ||
| 1566 |