Ansel 0.0
A darktable fork - bloat + design vision
Loading...
Searching...
No Matches
chromatic_adaptation.h
Go to the documentation of this file.
1/*
2 This file is part of darktable,
3 Copyright (C) 2020-2021, 2025 Aurélien PIERRE.
4 Copyright (C) 2021 Pascal Obry.
5 Copyright (C) 2021 Ralf Brown.
6 Copyright (C) 2022 Martin Bařinka.
7 Copyright (C) 2022 Sakari Kapanen.
8 Copyright (C) 2023 Luca Zulberti.
9
10 darktable is free software: you can redistribute it and/or modify
11 it under the terms of the GNU General Public License as published by
12 the Free Software Foundation, either version 3 of the License, or
13 (at your option) any later version.
14
15 darktable is distributed in the hope that it will be useful,
16 but WITHOUT ANY WARRANTY; without even the implied warranty of
17 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 GNU General Public License for more details.
19
20 You should have received a copy of the GNU General Public License
21 along with darktable. If not, see <http://www.gnu.org/licenses/>.
22*/
23
24#pragma once
25
26#include "common/math.h"
28
29typedef enum dt_adaptation_t
30{
31 DT_ADAPTATION_LINEAR_BRADFORD = 0, // $DESCRIPTION: "linear Bradford (ICC v4)"
32 DT_ADAPTATION_CAT16 = 1, // $DESCRIPTION: "CAT16 (CIECAM16)"
33 DT_ADAPTATION_FULL_BRADFORD = 2, // $DESCRIPTION: "non-linear Bradford"
34 DT_ADAPTATION_XYZ = 3, // $DESCRIPTION: "XYZ"
35 DT_ADAPTATION_RGB = 4, // $DESCRIPTION: "none (bypass)"
38
39
40// modified LMS cone response space for Bradford transform
41// explanation here : https://onlinelibrary.wiley.com/doi/pdf/10.1002/9781119021780.app3
42// but coeffs are wrong in the above, so they come from :
43// http://www2.cmp.uea.ac.uk/Research/compvis/Papers/FinSuss_COL00.pdf
44// At any time, ensure XYZ_to_LMS is the exact matrice inverse of LMS_to_XYZ
45const dt_colormatrix_t XYZ_to_Bradford_LMS = { { 0.8951f, 0.2664f, -0.1614f, 0.f },
46 { -0.7502f, 1.7135f, 0.0367f, 0.f },
47 { 0.0389f, -0.0685f, 1.0296f, 0.f } };
48
49const dt_colormatrix_t Bradford_LMS_to_XYZ = { { 0.9870f, -0.1471f, 0.1600f, 0.f },
50 { 0.4323f, 0.5184f, 0.0493f, 0.f },
51 { -0.0085f, 0.0400f, 0.9685f, 0.f } };
52
53static const dt_colormatrix_t XYZ_to_Bradford_LMS_transposed = { { 0.8951f, -0.7502f, 0.0389f, 0.f },
54 { 0.2664f, 1.7135f, -0.0685f, 0.f },
55 { -0.1614f, 0.0367f, 1.0296f, 0.f } };
56
57static const dt_colormatrix_t Bradford_LMS_to_XYZ_transposed = { { 0.9870f, 0.4323f, -0.0085f, 0.f },
58 { -0.1471f, 0.5184f, 0.0400f, 0.f },
59 { 0.1600f, 0.0493f, 0.9685f, 0.f } };
60
61static inline __attribute__((always_inline)) dt_aligned_pixel_simd_t
62convert_XYZ_to_bradford_LMS(const dt_aligned_pixel_simd_t XYZ)
63{
64 // Warning : needs XYZ normalized with Y - you need to downscale before
65 return dt_mat3x4_mul_vec4(XYZ,
66 dt_colormatrix_row_to_simd(XYZ_to_Bradford_LMS_transposed, 0),
67 dt_colormatrix_row_to_simd(XYZ_to_Bradford_LMS_transposed, 1),
68 dt_colormatrix_row_to_simd(XYZ_to_Bradford_LMS_transposed, 2));
69}
70
71static inline __attribute__((always_inline)) dt_aligned_pixel_simd_t
72convert_bradford_LMS_to_XYZ(const dt_aligned_pixel_simd_t LMS)
73{
74 // Warning : output XYZ normalized with Y - you need to upscale later
75 return dt_mat3x4_mul_vec4(LMS,
76 dt_colormatrix_row_to_simd(Bradford_LMS_to_XYZ_transposed, 0),
77 dt_colormatrix_row_to_simd(Bradford_LMS_to_XYZ_transposed, 1),
78 dt_colormatrix_row_to_simd(Bradford_LMS_to_XYZ_transposed, 2));
79}
80
81
82// modified LMS cone response for CAT16, from CIECAM16
83// reference : https://ntnuopen.ntnu.no/ntnu-xmlui/bitstream/handle/11250/2626317/CCIW-23.pdf?sequence=1
84// At any time, ensure XYZ_to_LMS is the exact matrice inverse of LMS_to_XYZ
85const dt_colormatrix_t XYZ_to_CAT16_LMS = { { 0.401288f, 0.650173f, -0.051461f, 0.f },
86 { -0.250268f, 1.204414f, 0.045854f, 0.f },
87 { -0.002079f, 0.048952f, 0.953127f, 0.f } };
88
89const dt_colormatrix_t CAT16_LMS_to_XYZ = { { 1.862068f, -1.011255f, 0.149187f, 0.f },
90 { 0.38752f , 0.621447f, -0.008974f, 0.f },
91 { -0.015841f, -0.034123f, 1.049964f, 0.f } };
92
93static const dt_colormatrix_t XYZ_to_CAT16_LMS_transposed = { { 0.401288f, -0.250268f, -0.002079f, 0.f },
94 { 0.650173f, 1.204414f, 0.048952f, 0.f },
95 { -0.051461f, 0.045854f, 0.953127f, 0.f } };
96
97static const dt_colormatrix_t CAT16_LMS_to_XYZ_transposed = { { 1.862068f, 0.38752f , -0.015841f, 0.f },
98 { -1.011255f, 0.621447f, -0.034123f, 0.f },
99 { 0.149187f, -0.008974f, 1.049964f, 0.f } };
100
101static inline __attribute__((always_inline)) dt_aligned_pixel_simd_t
102convert_XYZ_to_CAT16_LMS(const dt_aligned_pixel_simd_t XYZ)
103{
104 // Warning : needs XYZ normalized with Y - you need to downscale before
105 return dt_mat3x4_mul_vec4(XYZ,
106 dt_colormatrix_row_to_simd(XYZ_to_CAT16_LMS_transposed, 0),
107 dt_colormatrix_row_to_simd(XYZ_to_CAT16_LMS_transposed, 1),
108 dt_colormatrix_row_to_simd(XYZ_to_CAT16_LMS_transposed, 2));
109}
110
111static inline __attribute__((always_inline)) dt_aligned_pixel_simd_t
112convert_CAT16_LMS_to_XYZ(const dt_aligned_pixel_simd_t LMS)
113{
114 // Warning : output XYZ normalized with Y - you need to upscale later
115 return dt_mat3x4_mul_vec4(LMS,
116 dt_colormatrix_row_to_simd(CAT16_LMS_to_XYZ_transposed, 0),
117 dt_colormatrix_row_to_simd(CAT16_LMS_to_XYZ_transposed, 1),
118 dt_colormatrix_row_to_simd(CAT16_LMS_to_XYZ_transposed, 2));
119}
120
121
122static inline __attribute__((always_inline)) dt_aligned_pixel_simd_t
123convert_any_LMS_to_XYZ(const dt_aligned_pixel_simd_t LMS, const dt_adaptation_t kind)
124{
125 switch(kind)
126 {
135 default:
136 return LMS;
137 }
138}
139
140
141static inline __attribute__((always_inline)) dt_aligned_pixel_simd_t
142convert_any_XYZ_to_LMS(const dt_aligned_pixel_simd_t XYZ, const dt_adaptation_t kind)
143{
144 switch(kind)
145 {
154 default:
155 return XYZ;
156 }
157}
158
159
160#ifdef _OPENMP
161#pragma omp declare simd aligned(RGB, LMS:16) uniform(kind)
162#endif
163static inline void convert_any_LMS_to_RGB(const dt_aligned_pixel_t LMS, dt_aligned_pixel_t RGB, dt_adaptation_t kind)
164{
165 // helper function switching internally to the proper conversion
166 dt_aligned_pixel_t XYZ = { 0.f };
167 dt_store_simd_aligned(XYZ, convert_any_LMS_to_XYZ(dt_load_simd_aligned(LMS), kind));
168
169 // Fixme : convert to RGB display space instead of sRGB but first the display profile should be global in dt,
170 // not confined to colorout where it gets created/destroyed all the time.
171 dt_XYZ_to_Rec709_D65(XYZ, RGB);
172
173 // Handle gamut clipping
174 float max_RGB = fmaxf(fmaxf(RGB[0], RGB[1]), RGB[2]);
175 for(int c = 0; c < 3; c++) RGB[c] = fmaxf(RGB[c] / max_RGB, 0.f);
176
177}
178
179
180/* Bradford adaptations pre-computed for D50 and D65 outputs */
181
182static inline __attribute__((always_inline)) dt_aligned_pixel_simd_t
183bradford_adapt_D65(const dt_aligned_pixel_simd_t lms_in,
184 const dt_aligned_pixel_simd_t origin_illuminant,
185 const float p, const int full)
186{
187 static const dt_aligned_pixel_simd_t D65 = { 0.941238f, 1.040633f, 1.088932f, 0.f };
188 dt_aligned_pixel_simd_t temp = lms_in / origin_illuminant;
189 if(full) temp[2] = (temp[2] > 0.f) ? powf(temp[2], p) : temp[2];
190 return D65 * temp;
191}
192
193
194static inline __attribute__((always_inline)) dt_aligned_pixel_simd_t
195bradford_adapt_D50(const dt_aligned_pixel_simd_t lms_in,
196 const dt_aligned_pixel_simd_t origin_illuminant,
197 const float p, const int full)
198{
199 static const dt_aligned_pixel_simd_t D50 = { 0.996078f, 1.020646f, 0.818155f, 0.f };
200 dt_aligned_pixel_simd_t temp = lms_in / origin_illuminant;
201 if(full) temp[2] = (temp[2] > 0.f) ? powf(temp[2], p) : temp[2];
202 return D50 * temp;
203}
204
205
206/* CAT16 adaptations pre-computed for D50 and D65 outputs */
207
208static inline __attribute__((always_inline)) dt_aligned_pixel_simd_t
209CAT16_adapt_D65(const dt_aligned_pixel_simd_t lms_in,
210 const dt_aligned_pixel_simd_t origin_illuminant,
211 const float D, const int full)
212{
213 static const dt_aligned_pixel_simd_t D65 = { 0.97553267f, 1.01647859f, 1.0848344f, 0.f };
214 return full ? lms_in * D65 / origin_illuminant
215 : lms_in * (dt_simd_set1(D) * D65 / origin_illuminant + dt_simd_set1(1.f - D));
216}
217
218
219static inline __attribute__((always_inline)) dt_aligned_pixel_simd_t
220CAT16_adapt_D50(const dt_aligned_pixel_simd_t lms_in,
221 const dt_aligned_pixel_simd_t origin_illuminant,
222 const float D, const int full)
223{
224 static const dt_aligned_pixel_simd_t D50 = { 0.994535f, 1.000997f, 0.833036f, 0.f };
225 return full ? lms_in * D50 / origin_illuminant
226 : lms_in * (dt_simd_set1(D) * D50 / origin_illuminant + dt_simd_set1(1.f - D));
227}
228
229/* XYZ adaptations pre-computed for D50 and D65 outputs */
230
231static inline __attribute__((always_inline)) dt_aligned_pixel_simd_t
232XYZ_adapt_D65(const dt_aligned_pixel_simd_t lms_in,
233 const dt_aligned_pixel_simd_t origin_illuminant)
234{
235 static const dt_aligned_pixel_simd_t D65 = { 0.9504285453771807f, 1.0f, 1.0889003707981277f, 0.f };
236 return lms_in * D65 / origin_illuminant;
237}
238
239static inline __attribute__((always_inline)) dt_aligned_pixel_simd_t
240XYZ_adapt_D50(const dt_aligned_pixel_simd_t lms_in,
241 const dt_aligned_pixel_simd_t origin_illuminant)
242{
243 static const dt_aligned_pixel_simd_t D50 = { 0.9642119944211994f, 1.0f, 0.8251882845188288f, 0.f };
244 return lms_in * D50 / origin_illuminant;
245}
246
247/* Pre-solved matrices to adjust white point for triplets in CIE XYZ 1931 2° observer */
248
249const dt_colormatrix_t XYZ_D50_to_D65_CAT16
250 = { { 9.89466254e-01f, -4.00304626e-02f, 4.40530317e-02f, 0.f },
251 { -5.40518733e-03f, 1.00666069e+00f, -1.75551955e-03f, 0.f },
252 { -4.03920992e-04f, 1.50768030e-02f, 1.30210211e+00f, 0.f } };
253
255 = { { 0.95547342f, -0.02309845f, 0.06325924f, 0.f },
256 { -0.02836971f, 1.00999540f, 0.02104144f, 0.f },
257 { 0.01231401f, -0.02050765f, 1.33036593f, 0.f } };
258
260 = { { 1.01085433e+00f, 4.07086103e-02f, -3.41445825e-02f, 0.f },
261 { 5.42814201e-03f, 9.93581926e-01f, 1.15592039e-03f, 0.f },
262 { 2.50722468e-04f, -1.14918759e-02f, 7.67964947e-01f, 0.f } };
263
265 = { { 1.04792979f, 0.02294687f, -0.05019227f, 0.f },
266 { 0.02962781f, 0.99043443f, -0.0170738f, 0.f },
267 { -0.00924304f, 0.01505519f, 0.75187428f, 0.f } };
268
270 = { { 9.89466254e-01f, -5.40518733e-03f, -4.03920992e-04f, 0.f },
271 { -4.00304626e-02f, 1.00666069e+00f, 1.50768030e-02f, 0.f },
272 { 4.40530317e-02f, -1.75551955e-03f, 1.30210211e+00f, 0.f } };
273
275 = { { 1.01085433e+00f, 5.42814201e-03f, 2.50722468e-04f, 0.f },
276 { 4.07086103e-02f, 9.93581926e-01f, -1.14918759e-02f, 0.f },
277 { -3.41445825e-02f, 1.15592039e-03f, 7.67964947e-01f, 0.f } };
278
279#ifdef _OPENMP
280#pragma omp declare simd aligned(XYZ_in, XYZ_out:16)
281#endif
282static inline void XYZ_D50_to_D65(const dt_aligned_pixel_t XYZ_in, dt_aligned_pixel_t XYZ_out)
283{
285}
286
287#ifdef _OPENMP
288#pragma omp declare simd aligned(XYZ_in, XYZ_out:16)
289#endif
290static inline void XYZ_D65_to_D50(const dt_aligned_pixel_t XYZ_in, dt_aligned_pixel_t XYZ_out)
291{
293}
294
295/* Helper function to directly chroma-adapt a pixel in CIE XYZ 1931 2° */
296
297static inline __attribute__((always_inline)) dt_aligned_pixel_simd_t
298_downscale_vector_simd(const dt_aligned_pixel_simd_t vector, const float scaling)
299{
300 const int valid = (scaling > NORM_MIN) && !isnan(scaling);
301 return vector / dt_simd_set1(valid ? (scaling + NORM_MIN) : NORM_MIN);
302}
303
304static inline __attribute__((always_inline)) dt_aligned_pixel_simd_t
305_upscale_vector_simd(const dt_aligned_pixel_simd_t vector, const float scaling)
306{
307 const int valid = (scaling > NORM_MIN) && !isnan(scaling);
308 return vector * dt_simd_set1(valid ? (scaling + NORM_MIN) : NORM_MIN);
309}
310
311static inline __attribute__((always_inline)) dt_aligned_pixel_simd_t
312chroma_adapt_pixel(const dt_aligned_pixel_simd_t in,
313 const dt_aligned_pixel_simd_t illuminant,
314 const dt_adaptation_t adaptation, const float p)
315{
316
317 /* WE START IN XYZ */
318 const float Y = in[1];
319
321 {
324 _upscale_vector_simd(bradford_adapt_D50(
325 _downscale_vector_simd(convert_XYZ_to_bradford_LMS(in), Y),
326 illuminant, p, TRUE),
327 Y));
330 _upscale_vector_simd(bradford_adapt_D50(
331 _downscale_vector_simd(convert_XYZ_to_bradford_LMS(in), Y),
332 illuminant, p, FALSE),
333 Y));
336 _upscale_vector_simd(CAT16_adapt_D50(
337 _downscale_vector_simd(convert_XYZ_to_CAT16_LMS(in), Y),
338 illuminant, 1.0f, TRUE),
339 Y));
341 return _upscale_vector_simd(XYZ_adapt_D50(_downscale_vector_simd(in, Y), illuminant), Y);
344 default:
345 return in;
346 }
347}
348
349/* Helper to get the D50 white point coordinates in LMS spaces */
350static inline void convert_D50_to_LMS(const dt_adaptation_t adaptation, dt_aligned_pixel_t D50)
351{
352 switch(adaptation)
353 {
356 {
357 dt_store_simd_aligned(D50, (dt_aligned_pixel_simd_t){ 0.996078f, 1.020646f, 0.818155f, 0.f });
358 break;
359 }
361 {
362 dt_store_simd_aligned(D50, (dt_aligned_pixel_simd_t){ 0.994535f, 1.000997f, 0.833036f, 0.f });
363 break;
364 }
366 {
367 dt_store_simd_aligned(D50, (dt_aligned_pixel_simd_t){ 0.9642119944211994f, 1.0f, 0.8251882845188288f, 0.f });
368 break;
369 }
372 default:
373 {
374 dt_store_simd_aligned(D50, (dt_aligned_pixel_simd_t){ 1.f, 1.f, 1.f, 0.f });
375 break;
376 }
377 }
378}
379// clang-format off
380// modelines: These editor modelines have been set for all relevant files by tools/update_modelines.py
381// vim: shiftwidth=2 expandtab tabstop=2 cindent
382// kate: tab-indents: off; indent-width 2; replace-tabs on; indent-mode cstyle; remove-trailing-spaces modified;
383// clang-format on
#define TRUE
Definition ashift_lsd.c:162
#define FALSE
Definition ashift_lsd.c:158
static void XYZ_D50_to_D65(const dt_aligned_pixel_t XYZ_in, dt_aligned_pixel_t XYZ_out)
Definition chromatic_adaptation.h:282
static const dt_aligned_pixel_simd_t const dt_adaptation_t adaptation
Definition chromatic_adaptation.h:314
const dt_colormatrix_t XYZ_D50_to_D65_Bradford
Definition chromatic_adaptation.h:255
const dt_colormatrix_t CAT16_LMS_to_XYZ
Definition chromatic_adaptation.h:89
static void XYZ_D65_to_D50(const dt_aligned_pixel_t XYZ_in, dt_aligned_pixel_t XYZ_out)
Definition chromatic_adaptation.h:290
static const dt_adaptation_t kind
Definition chromatic_adaptation.h:124
const dt_colormatrix_t XYZ_D65_to_D50_CAT16
Definition chromatic_adaptation.h:260
const dt_colormatrix_t XYZ_to_CAT16_LMS
Definition chromatic_adaptation.h:85
static void convert_D50_to_LMS(const dt_adaptation_t adaptation, dt_aligned_pixel_t D50)
Definition chromatic_adaptation.h:350
static const dt_colormatrix_t XYZ_to_CAT16_LMS_transposed
Definition chromatic_adaptation.h:93
dt_adaptation_t
Definition chromatic_adaptation.h:30
@ DT_ADAPTATION_LAST
Definition chromatic_adaptation.h:36
@ DT_ADAPTATION_FULL_BRADFORD
Definition chromatic_adaptation.h:33
@ DT_ADAPTATION_XYZ
Definition chromatic_adaptation.h:34
@ DT_ADAPTATION_CAT16
Definition chromatic_adaptation.h:32
@ DT_ADAPTATION_RGB
Definition chromatic_adaptation.h:35
@ DT_ADAPTATION_LINEAR_BRADFORD
Definition chromatic_adaptation.h:31
static const dt_colormatrix_t XYZ_to_Bradford_LMS_transposed
Definition chromatic_adaptation.h:53
static const dt_colormatrix_t XYZ_D65_to_D50_CAT16_transposed
Definition chromatic_adaptation.h:275
static const dt_colormatrix_t XYZ_D50_to_D65_CAT16_transposed
Definition chromatic_adaptation.h:270
static const dt_aligned_pixel_simd_t const dt_adaptation_t const float p
Definition chromatic_adaptation.h:315
const dt_colormatrix_t XYZ_to_Bradford_LMS
Definition chromatic_adaptation.h:45
const dt_colormatrix_t Bradford_LMS_to_XYZ
Definition chromatic_adaptation.h:49
static const float scaling
Definition chromatic_adaptation.h:299
static const dt_colormatrix_t CAT16_LMS_to_XYZ_transposed
Definition chromatic_adaptation.h:97
const dt_colormatrix_t XYZ_D65_to_D50_Bradford
Definition chromatic_adaptation.h:265
static const dt_colormatrix_t Bradford_LMS_to_XYZ_transposed
Definition chromatic_adaptation.h:57
return vector dt_simd_set1(valid ?(scaling+NORM_MIN) :NORM_MIN)
static const dt_aligned_pixel_simd_t illuminant
Definition chromatic_adaptation.h:313
static void CAT16_adapt_D50(float4 *lms_in, const float4 origin_illuminant, const float D, const int full)
Definition colorspace.h:722
static float4 convert_XYZ_to_bradford_LMS(const float4 XYZ)
Definition colorspace.h:657
static float4 convert_XYZ_to_CAT16_LMS(const float4 XYZ)
Definition colorspace.h:677
static void bradford_adapt_D50(float4 *lms_in, const float4 origin_illuminant, const float p, const int full)
Definition colorspace.h:697
static void XYZ_adapt_D50(float4 *lms_in, const float4 origin_illuminant)
Definition colorspace.h:737
static float4 convert_CAT16_LMS_to_XYZ(const float4 LMS)
Definition colorspace.h:687
static float4 convert_bradford_LMS_to_XYZ(const float4 LMS)
Definition colorspace.h:667
dt_aligned_pixel_t LMS
Definition colorspaces_inline_conversions.h:952
const float c
Definition colorspaces_inline_conversions.h:1365
dt_apply_transposed_color_matrix(XYZ, xyz_to_srgb_matrix_transposed, sRGB)
static dt_aligned_pixel_t XYZ
Definition colorspaces_inline_conversions.h:252
dt_store_simd_aligned(out, dt_mat3x4_mul_vec4(vin, dt_colormatrix_row_to_simd(matrix, 0), dt_colormatrix_row_to_simd(matrix, 1), dt_colormatrix_row_to_simd(matrix, 2)))
static dt_aligned_pixel_t RGB
Definition colorspaces_inline_conversions.h:509
return dt_load_simd_aligned(JCH)
float dt_aligned_pixel_simd_t __attribute__((vector_size(16), aligned(16)))
Multi-tap smudge source sample with directional jitter.
Definition darktable.h:448
#define NORM_MIN
Definition math.h:35
float DT_ALIGNED_ARRAY dt_colormatrix_t[4][4]
Definition matrices.h:33