Ansel 0.0
A darktable fork - bloat + design vision
Loading...
Searching...
No Matches
dual.c
Go to the documentation of this file.
1/*
2 This file is part of darktable,
3 Copyright (C) 2021-2022 Hanno Schwalm.
4 Copyright (C) 2021 luzpaz.
5 Copyright (C) 2021 Marco.
6 Copyright (C) 2021 Pascal Obry.
7 Copyright (C) 2021 Ralf Brown.
8 Copyright (C) 2022 Martin Bařinka.
9 Copyright (C) 2023, 2026 Aurélien PIERRE.
10
11 darktable is free software: you can redistribute it and/or modify
12 it under the terms of the GNU General Public License as published by
13 the Free Software Foundation, either version 3 of the License, or
14 (at your option) any later version.
15
16 darktable is distributed in the hope that it will be useful,
17 but WITHOUT ANY WARRANTY; without even the implied warranty of
18 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19 GNU General Public License for more details.
20
21 You should have received a copy of the GNU General Public License
22 along with darktable. If not, see <http://www.gnu.org/licenses/>.
23*/
24
25/*
26 Dual demosaicing has been implemented by Ingo Weyrich <heckflosse67@gmx.de> for
27 rawtherapee under GNU General Public License Version 3
28 and has been modified to work for darktable by Hanno Schwalm (hanno@schwalm-bremen.de).
29 Also the code for dt_masks_blur_9x9 has been taken from rawtherapee capturesharpening,
30 implemented also by Ingo Weyrich.
31*/
32
33
34static float slider2contrast(float slider)
35{
36 return 0.005f * powf(slider, 1.1f);
37}
39static int dual_demosaic(const dt_dev_pixelpipe_t *pipe, const dt_dev_pixelpipe_iop_t *piece,
40 float *const restrict rgb_data, const float *const restrict raw_data,
41 dt_iop_roi_t *const roi_out, const dt_iop_roi_t *const roi_in,
42 const uint32_t filters, const uint8_t (*const xtrans)[6],
43 const gboolean dual_mask, float dual_threshold)
44{
45 const int width = roi_in->width;
46 const int height = roi_in->height;
47 if((width < 16) || (height < 16)) return 0;
48
49 // If the threshold is zero and we don't want to see the blend mask we don't do anything
50 if(dual_threshold <= 0.0f) return 0;
51
52 float *blend = dt_pixelpipe_cache_alloc_align_float((size_t) width * height, pipe);
53 float *tmp = dt_pixelpipe_cache_alloc_align_float((size_t) width * height, pipe);
54 float *vng_image = dt_pixelpipe_cache_alloc_align_float((size_t) 4 * width * height, pipe);
55 if(IS_NULL_PTR(blend) || IS_NULL_PTR(tmp) || IS_NULL_PTR(vng_image))
56 {
60 dt_control_log(_("[dual demosaic] can't allocate internal buffers"));
61 return 1;
62 }
63 const gboolean info = ((darktable.unmuted & (DT_DEBUG_DEMOSAIC | DT_DEBUG_PERF))
64 && (pipe->type == DT_DEV_PIXELPIPE_FULL));
65
66 if(vng_interpolate(vng_image, raw_data, roi_out, roi_in, filters, xtrans, FALSE))
67 {
71 return 1;
72 }
73 color_smoothing(vng_image, roi_out, 2);
74
75 dt_times_t start_blend = { 0 }, end_blend = { 0 };
76 if(info) dt_get_times(&start_blend);
77
78 const float contrastf = slider2contrast(dual_threshold);
79
80 dt_masks_calc_rawdetail_mask(rgb_data, blend, tmp, width, height, piece->dsc_in.temperature.coeffs);
81 dt_masks_calc_detail_mask(blend, blend, tmp, width, height, contrastf, TRUE);
82
83 if(dual_mask)
84 {
86 __OMP_FOR_SIMD__(aligned(blend, vng_image, rgb_data : 64))
87 for(int idx = 0; idx < width * height; idx++)
88 {
89 for(int c = 0; c < 4; c++)
90 rgb_data[idx * 4 + c] = blend[idx];
91 }
92 }
93 else
94 {
95 __OMP_FOR_SIMD__(aligned(blend, vng_image, rgb_data : 64))
96 for(int idx = 0; idx < width * height; idx++)
97 {
98 const int oidx = 4 * idx;
99 for(int c = 0; c < 4; c++)
100 rgb_data[oidx + c] = intp(blend[idx], rgb_data[oidx + c], vng_image[oidx + c]);
101 }
102 }
103 if(info)
104 {
105 dt_get_times(&end_blend);
106 fprintf(stderr," [demosaic] CPU dual blending %.4f secs (%.4f CPU)\n", end_blend.clock - start_blend.clock, end_blend.user - start_blend.user);
107 }
111 return 0;
112}
113
114#ifdef HAVE_OPENCL
115gboolean dual_demosaic_cl(struct dt_iop_module_t *self, const dt_dev_pixelpipe_t *pipe,
116 const dt_dev_pixelpipe_iop_t *piece, cl_mem detail, cl_mem blend,
117 cl_mem high_image, cl_mem low_image, cl_mem out, const int width,
118 const int height, const int showmask)
119{
120 const int devid = pipe->devid;
123
124 const float contrastf = slider2contrast(data->dual_thrs);
125 if(showmask)
127
128 {
129 size_t sizes[3] = { ROUNDUPDWD(width, devid), ROUNDUPDHT(height, devid), 1 };
130 const dt_aligned_pixel_t wb = { piece->dsc_in.temperature.coeffs[0], piece->dsc_in.temperature.coeffs[1],
131 piece->dsc_in.temperature.coeffs[2] };
133 dt_opencl_set_kernel_arg(devid, kernel, 0, sizeof(cl_mem), &detail);
134 dt_opencl_set_kernel_arg(devid, kernel, 1, sizeof(cl_mem), &high_image);
135 dt_opencl_set_kernel_arg(devid, kernel, 2, sizeof(int), &width);
136 dt_opencl_set_kernel_arg(devid, kernel, 3, sizeof(int), &height);
137 dt_opencl_set_kernel_arg(devid, kernel, 4, sizeof(float), &wb[0]);
138 dt_opencl_set_kernel_arg(devid, kernel, 5, sizeof(float), &wb[1]);
139 dt_opencl_set_kernel_arg(devid, kernel, 6, sizeof(float), &wb[2]);
140 const int err = dt_opencl_enqueue_kernel_2d(devid, kernel, sizes);
141 if(err != CL_SUCCESS) return FALSE;
142 }
143
144 {
145 size_t sizes[3] = { ROUNDUPDWD(width, devid), ROUNDUPDHT(height, devid), 1 };
147 dt_opencl_set_kernel_arg(devid, kernel, 0, sizeof(cl_mem), &detail);
148 dt_opencl_set_kernel_arg(devid, kernel, 1, sizeof(cl_mem), &blend);
149 dt_opencl_set_kernel_arg(devid, kernel, 2, sizeof(int), &width);
150 dt_opencl_set_kernel_arg(devid, kernel, 3, sizeof(int), &height);
151 const int err = dt_opencl_enqueue_kernel_2d(devid, kernel, sizes);
152 if(err != CL_SUCCESS) return FALSE;
153 }
154
155 {
156 const int flag = 1;
157 size_t sizes[3] = { ROUNDUPDWD(width, devid), ROUNDUPDHT(height, devid), 1 };
159 dt_opencl_set_kernel_arg(devid, kernel, 0, sizeof(cl_mem), &blend);
160 dt_opencl_set_kernel_arg(devid, kernel, 1, sizeof(cl_mem), &detail);
161 dt_opencl_set_kernel_arg(devid, kernel, 2, sizeof(int), &width);
162 dt_opencl_set_kernel_arg(devid, kernel, 3, sizeof(int), &height);
163 dt_opencl_set_kernel_arg(devid, kernel, 4, sizeof(float), &contrastf);
164 dt_opencl_set_kernel_arg(devid, kernel, 5, sizeof(int), &flag);
165 const int err = dt_opencl_enqueue_kernel_2d(devid, kernel, sizes);
166 if(err != CL_SUCCESS) return FALSE;
167 }
168
169 {
170 float blurmat[13];
171 dt_masks_blur_9x9_coeff(blurmat, 2.0f);
172 cl_mem dev_blurmat = NULL;
173 dev_blurmat = dt_opencl_copy_host_to_device_constant(devid, sizeof(float) * 13, blurmat);
174 if(!IS_NULL_PTR(dev_blurmat))
175 {
176 size_t sizes[3] = { ROUNDUPDWD(width, devid), ROUNDUPDHT(height, devid), 1 };
177 const int clkernel = darktable.opencl->blendop->kernel_mask_blur;
178 dt_opencl_set_kernel_arg(devid, clkernel, 0, sizeof(cl_mem), &detail);
179 dt_opencl_set_kernel_arg(devid, clkernel, 1, sizeof(cl_mem), &blend);
180 dt_opencl_set_kernel_arg(devid, clkernel, 2, sizeof(int), &width);
181 dt_opencl_set_kernel_arg(devid, clkernel, 3, sizeof(int), &height);
182 dt_opencl_set_kernel_arg(devid, clkernel, 4, sizeof(cl_mem), (void *) &dev_blurmat);
183 const int err = dt_opencl_enqueue_kernel_2d(devid, clkernel, sizes);
184 dt_opencl_release_mem_object(dev_blurmat);
185 if(err != CL_SUCCESS) return FALSE;
186 }
187 else
188 {
189 dt_opencl_release_mem_object(dev_blurmat);
190 return FALSE;
191 }
192 }
193
194 {
195 size_t sizes[3] = { ROUNDUPDWD(width, devid), ROUNDUPDHT(height, devid), 1 };
196 dt_opencl_set_kernel_arg(devid, gd->kernel_write_blended_dual, 0, sizeof(cl_mem), &high_image);
197 dt_opencl_set_kernel_arg(devid, gd->kernel_write_blended_dual, 1, sizeof(cl_mem), &low_image);
198 dt_opencl_set_kernel_arg(devid, gd->kernel_write_blended_dual, 2, sizeof(cl_mem), &out);
199 dt_opencl_set_kernel_arg(devid, gd->kernel_write_blended_dual, 3, sizeof(int), &width);
200 dt_opencl_set_kernel_arg(devid, gd->kernel_write_blended_dual, 4, sizeof(int), &height);
201 dt_opencl_set_kernel_arg(devid, gd->kernel_write_blended_dual, 5, sizeof(cl_mem), &blend);
202 dt_opencl_set_kernel_arg(devid, gd->kernel_write_blended_dual, 6, sizeof(int), &showmask);
203 const int err = dt_opencl_enqueue_kernel_2d(devid, gd->kernel_write_blended_dual, sizes);
204 if(err != CL_SUCCESS) return FALSE;
205 }
206
207 return TRUE;
208}
209#endif
210
211// clang-format off
212// modelines: These editor modelines have been set for all relevant files by tools/update_modelines.py
213// vim: shiftwidth=2 expandtab tabstop=2 cindent
214// kate: tab-indents: off; indent-width 2; replace-tabs on; indent-mode cstyle; remove-trailing-spaces modified;
215// clang-format on
#define TRUE
Definition ashift_lsd.c:162
#define FALSE
Definition ashift_lsd.c:158
static __DT_CLONE_TARGETS__ void color_smoothing(float *out, const dt_iop_roi_t *const roi_out, const int num_passes)
Definition basic.c:192
int width
Definition bilateral.h:1
int height
Definition bilateral.h:1
static float intp(const float a, const float b, const float c)
Definition cacorrect.c:180
static float kernel(const float *x, const float *y)
Definition colorchecker.c:469
const dt_colormatrix_t dt_aligned_pixel_t out
Definition colorspaces_inline_conversions.h:42
void dt_control_log(const char *msg,...)
Definition control.c:530
darktable_t darktable
Definition darktable.c:173
@ DT_DEBUG_DEMOSAIC
Definition darktable.h:736
@ DT_DEBUG_PERF
Definition darktable.h:718
static void dt_get_times(dt_times_t *t)
Definition darktable.h:920
#define dt_pixelpipe_cache_free_align(mem)
Definition darktable.h:453
#define dt_pixelpipe_cache_alloc_align_float(pixels, pipe)
Definition darktable.h:442
#define __DT_CLONE_TARGETS__
Definition darktable.h:367
#define __OMP_FOR_SIMD__(...)
Definition darktable.h:260
#define IS_NULL_PTR(p)
C is way too permissive with !=, == and if(var) checks, which can mean too many things depending on w...
Definition darktable.h:281
@ DT_DEV_PIXELPIPE_DISPLAY_PASSTHRU
Definition develop.h:136
static __DT_CLONE_TARGETS__ int dual_demosaic(const dt_dev_pixelpipe_t *pipe, const dt_dev_pixelpipe_iop_t *piece, float *const restrict rgb_data, const float *const restrict raw_data, dt_iop_roi_t *const roi_out, const dt_iop_roi_t *const roi_in, const uint32_t filters, const uint8_t(*const xtrans)[6], const gboolean dual_mask, float dual_threshold)
Definition dual.c:39
static float slider2contrast(float slider)
Definition dual.c:34
gboolean dual_demosaic_cl(struct dt_iop_module_t *self, const dt_dev_pixelpipe_t *pipe, const dt_dev_pixelpipe_iop_t *piece, cl_mem detail, cl_mem blend, cl_mem high_image, cl_mem low_image, cl_mem out, const int width, const int height, const int showmask)
Definition dual.c:115
const char flag
Definition image.h:218
void dt_masks_calc_detail_mask(float *const src, float *const out, float *const tmp, const int width, const int height, const float threshold, const gboolean detail)
void dt_masks_blur_9x9_coeff(float *coeffs, const float sigma)
Definition detail.c:159
void dt_masks_calc_rawdetail_mask(float *const src, float *const out, float *const tmp, const int width, const int height, const dt_aligned_pixel_t wb)
float dt_aligned_pixel_t[4]
Definition noiseprofile.c:28
int dt_opencl_enqueue_kernel_2d(const int dev, const int kernel, const size_t *sizes)
Definition opencl.c:1951
void * dt_opencl_copy_host_to_device_constant(const int devid, const size_t size, void *host)
Definition opencl.c:2147
int dt_opencl_set_kernel_arg(const int dev, const int kernel, const int num, const size_t size, const void *arg)
Definition opencl.c:1942
void dt_opencl_release_mem_object(cl_mem mem)
Definition opencl.c:2198
#define ROUNDUPDHT(a, b)
Definition opencl.h:82
#define ROUNDUPDWD(a, b)
Definition opencl.h:81
@ DT_DEV_PIXELPIPE_FULL
Definition pixelpipe.h:39
struct dt_opencl_t * opencl
Definition darktable.h:784
int32_t unmuted
Definition darktable.h:759
int kernel_calc_blend
Definition blend.h:257
int kernel_calc_Y0_mask
Definition blend.h:253
int kernel_mask_blur
Definition blend.h:258
int kernel_calc_scharr_mask
Definition blend.h:254
Definition pixelpipe_hb.h:96
dt_iop_buffer_dsc_t dsc_in
Definition pixelpipe_hb.h:142
struct dt_iop_module_t *void * data
Definition pixelpipe_hb.h:97
Definition pixelpipe_hb.h:218
dt_dev_pixelpipe_type_t type
Definition pixelpipe_hb.h:300
int devid
Definition pixelpipe_hb.h:308
dt_aligned_pixel_t coeffs
Definition format.h:81
struct dt_iop_buffer_dsc_t::@17 temperature
Definition demosaic.c:228
float dual_thrs
Definition demosaic.c:235
Definition demosaic.c:162
int kernel_write_blended_dual
Definition demosaic.c:221
Definition imageop.h:246
dt_iop_global_data_t * global_data
Definition imageop.h:316
Region of interest passed through the pixelpipe.
Definition imageop.h:72
int width
Definition imageop.h:73
int height
Definition imageop.h:73
struct dt_blendop_cl_global_t * blendop
Definition opencl.h:238
Definition darktable.h:840
double clock
Definition darktable.h:841
double user
Definition darktable.h:842
static __DT_CLONE_TARGETS__ int vng_interpolate(float *out, const float *const in, const dt_iop_roi_t *const roi_out, const dt_iop_roi_t *const roi_in, const uint32_t filters, const uint8_t(*const xtrans)[6], const int only_vng_linear)
Definition vng.c:34