/* This file is part of darktable, copyright (c) 2013 ulrich pegelow. darktable is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. darktable is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with darktable. If not, see . */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include "bauhaus/bauhaus.h" #include "common/debug.h" #include "common/opencl.h" #include "control/control.h" #include "develop/develop.h" #include "develop/imageop.h" #include "develop/imageop_math.h" #include "develop/tiling.h" #include "gui/accelerators.h" #include "gui/gtk.h" #include "gui/presets.h" #include "iop/iop_api.h" #include #include #include #include #include #include DT_MODULE_INTROSPECTION(1, dt_iop_colisa_params_t) typedef struct dt_iop_colisa_params_t { float contrast; float brightness; float saturation; } dt_iop_colisa_params_t; typedef struct dt_iop_colisa_gui_data_t { GtkWidget *contrast; GtkWidget *brightness; GtkWidget *saturation; } dt_iop_colisa_gui_data_t; typedef struct dt_iop_colisa_data_t { float contrast; float brightness; float saturation; float ctable[0x10000]; // precomputed look-up table for contrast curve float cunbounded_coeffs[3]; // approximation for extrapolation of contrast curve float ltable[0x10000]; // precomputed look-up table for brightness curve float lunbounded_coeffs[3]; // approximation for extrapolation of brightness curve } dt_iop_colisa_data_t; typedef struct dt_iop_colisa_global_data_t { int kernel_colisa; } dt_iop_colisa_global_data_t; const char *name() { return _("contrast brightness saturation"); } int flags() { return IOP_FLAGS_INCLUDE_IN_STYLES | IOP_FLAGS_SUPPORTS_BLENDING | IOP_FLAGS_ALLOW_TILING; } int groups() { return IOP_GROUP_BASIC; } void init_key_accels(dt_iop_module_so_t *self) { dt_accel_register_slider_iop(self, FALSE, NC_("accel", "contrast")); dt_accel_register_slider_iop(self, FALSE, NC_("accel", "brightness")); dt_accel_register_slider_iop(self, FALSE, NC_("accel", "saturation")); } void connect_key_accels(dt_iop_module_t *self) { dt_iop_colisa_gui_data_t *g = (dt_iop_colisa_gui_data_t *)self->gui_data; dt_accel_connect_slider_iop(self, "contrast", GTK_WIDGET(g->contrast)); dt_accel_connect_slider_iop(self, "brightness", GTK_WIDGET(g->brightness)); dt_accel_connect_slider_iop(self, "saturation", GTK_WIDGET(g->saturation)); } #ifdef HAVE_OPENCL int process_cl(struct dt_iop_module_t *self, dt_dev_pixelpipe_iop_t *piece, cl_mem dev_in, cl_mem dev_out, const dt_iop_roi_t *const roi_in, const dt_iop_roi_t *const roi_out) { dt_iop_colisa_data_t *d = (dt_iop_colisa_data_t *)piece->data; dt_iop_colisa_global_data_t *gd = (dt_iop_colisa_global_data_t *)self->data; cl_int err = -999; const int devid = piece->pipe->devid; const int width = roi_in->width; const int height = roi_in->height; const float saturation = d->saturation; cl_mem dev_cm = NULL; cl_mem dev_ccoeffs = NULL; cl_mem dev_lm = NULL; cl_mem dev_lcoeffs = NULL; dev_cm = dt_opencl_copy_host_to_device(devid, d->ctable, 256, 256, sizeof(float)); if(dev_cm == NULL) goto error; dev_ccoeffs = dt_opencl_copy_host_to_device_constant(devid, sizeof(float) * 3, d->cunbounded_coeffs); if(dev_ccoeffs == NULL) goto error; dev_lm = dt_opencl_copy_host_to_device(devid, d->ltable, 256, 256, sizeof(float)); if(dev_lm == NULL) goto error; dev_lcoeffs = dt_opencl_copy_host_to_device_constant(devid, sizeof(float) * 3, d->lunbounded_coeffs); if(dev_lcoeffs == NULL) goto error; size_t sizes[3]; sizes[0] = ROUNDUPWD(width); sizes[1] = ROUNDUPWD(height); sizes[2] = 1; dt_opencl_set_kernel_arg(devid, gd->kernel_colisa, 0, sizeof(cl_mem), (void *)&dev_in); dt_opencl_set_kernel_arg(devid, gd->kernel_colisa, 1, sizeof(cl_mem), (void *)&dev_out); dt_opencl_set_kernel_arg(devid, gd->kernel_colisa, 2, sizeof(int), (void *)&width); dt_opencl_set_kernel_arg(devid, gd->kernel_colisa, 3, sizeof(int), (void *)&height); dt_opencl_set_kernel_arg(devid, gd->kernel_colisa, 4, sizeof(float), (void *)&saturation); dt_opencl_set_kernel_arg(devid, gd->kernel_colisa, 5, sizeof(cl_mem), (void *)&dev_cm); dt_opencl_set_kernel_arg(devid, gd->kernel_colisa, 6, sizeof(cl_mem), (void *)&dev_ccoeffs); dt_opencl_set_kernel_arg(devid, gd->kernel_colisa, 7, sizeof(cl_mem), (void *)&dev_lm); dt_opencl_set_kernel_arg(devid, gd->kernel_colisa, 8, sizeof(cl_mem), (void *)&dev_lcoeffs); err = dt_opencl_enqueue_kernel_2d(devid, gd->kernel_colisa, sizes); if(err != CL_SUCCESS) goto error; dt_opencl_release_mem_object(dev_lcoeffs); dt_opencl_release_mem_object(dev_lm); dt_opencl_release_mem_object(dev_ccoeffs); dt_opencl_release_mem_object(dev_cm); return TRUE; error: dt_opencl_release_mem_object(dev_lcoeffs); dt_opencl_release_mem_object(dev_lm); dt_opencl_release_mem_object(dev_ccoeffs); dt_opencl_release_mem_object(dev_cm); dt_print(DT_DEBUG_OPENCL, "[opencl_colisa] couldn't enqueue kernel! %d\n", err); return FALSE; } #endif void process(struct dt_iop_module_t *self, dt_dev_pixelpipe_iop_t *piece, const void *const ivoid, void *const ovoid, const dt_iop_roi_t *const roi_in, const dt_iop_roi_t *const roi_out) { dt_iop_colisa_data_t *data = (dt_iop_colisa_data_t *)piece->data; float *in = (float *)ivoid; float *out = (float *)ovoid; const int width = roi_in->width; const int height = roi_in->height; const int ch = piece->colors; #ifdef _OPENMP #pragma omp parallel for default(none) shared(in, out, data) schedule(static) #endif for(size_t k = 0; k < (size_t)width * height; k++) { float L = (in[k * ch + 0] < 100.0f) ? data->ctable[CLAMP((int)(in[k * ch + 0] / 100.0f * 0x10000ul), 0, 0xffff)] : dt_iop_eval_exp(data->cunbounded_coeffs, in[k * ch + 0] / 100.0f); out[k * ch + 0] = (L < 100.0f) ? data->ltable[CLAMP((int)(L / 100.0f * 0x10000ul), 0, 0xffff)] : dt_iop_eval_exp(data->lunbounded_coeffs, L / 100.0f); out[k * ch + 1] = in[k * ch + 1] * data->saturation; out[k * ch + 2] = in[k * ch + 2] * data->saturation; out[k * ch + 3] = in[k * ch + 3]; } } static void contrast_callback(GtkWidget *slider, gpointer user_data) { dt_iop_module_t *self = (dt_iop_module_t *)user_data; if(self->dt->gui->reset) return; dt_iop_colisa_params_t *p = (dt_iop_colisa_params_t *)self->params; p->contrast = dt_bauhaus_slider_get(slider); dt_dev_add_history_item(darktable.develop, self, TRUE); } static void brightness_callback(GtkWidget *slider, gpointer user_data) { dt_iop_module_t *self = (dt_iop_module_t *)user_data; if(self->dt->gui->reset) return; dt_iop_colisa_params_t *p = (dt_iop_colisa_params_t *)self->params; p->brightness = dt_bauhaus_slider_get(slider); dt_dev_add_history_item(darktable.develop, self, TRUE); } static void saturation_callback(GtkWidget *slider, gpointer user_data) { dt_iop_module_t *self = (dt_iop_module_t *)user_data; if(self->dt->gui->reset) return; dt_iop_colisa_params_t *p = (dt_iop_colisa_params_t *)self->params; p->saturation = dt_bauhaus_slider_get(slider); dt_dev_add_history_item(darktable.develop, self, TRUE); } void commit_params(struct dt_iop_module_t *self, dt_iop_params_t *p1, dt_dev_pixelpipe_t *pipe, dt_dev_pixelpipe_iop_t *piece) { dt_iop_colisa_params_t *p = (dt_iop_colisa_params_t *)p1; dt_iop_colisa_data_t *d = (dt_iop_colisa_data_t *)piece->data; d->contrast = p->contrast + 1.0f; // rescale from [-1;+1] to [0;+2] (zero meaning no contrast -> gray plane) d->brightness = p->brightness * 2.0f; // rescale from [-1;+1] to [-2;+2] d->saturation = p->saturation + 1.0f; // rescale from [-1;+1] to [0;+2] (zero meaning no saturation -> b&w) // generate precomputed contrast curve if(d->contrast <= 1.0f) { // linear curve for d->contrast below 1 #ifdef _OPENMP #pragma omp parallel for default(none) shared(d) schedule(static) #endif for(int k = 0; k < 0x10000; k++) d->ctable[k] = d->contrast * (100.0f * k / 0x10000 - 50.0f) + 50.0f; } else { // sigmoidal curve for d->contrast above 1 const float boost = 20.0f; const float contrastm1sq = boost * (d->contrast - 1.0f) * (d->contrast - 1.0f); const float contrastscale = sqrt(1.0f + contrastm1sq); #ifdef _OPENMP #pragma omp parallel for default(none) shared(d) schedule(static) #endif for(int k = 0; k < 0x10000; k++) { float kx2m1 = 2.0f * (float)k / 0x10000 - 1.0f; d->ctable[k] = 50.0f * (contrastscale * kx2m1 / sqrtf(1.0f + contrastm1sq * kx2m1 * kx2m1) + 1.0f); } } // now the extrapolation stuff for the contrast curve: const float xc[4] = { 0.7f, 0.8f, 0.9f, 1.0f }; const float yc[4] = { d->ctable[CLAMP((int)(xc[0] * 0x10000ul), 0, 0xffff)], d->ctable[CLAMP((int)(xc[1] * 0x10000ul), 0, 0xffff)], d->ctable[CLAMP((int)(xc[2] * 0x10000ul), 0, 0xffff)], d->ctable[CLAMP((int)(xc[3] * 0x10000ul), 0, 0xffff)] }; dt_iop_estimate_exp(xc, yc, 4, d->cunbounded_coeffs); // generate precomputed brightness curve const float gamma = (d->brightness >= 0.0f) ? 1.0f / (1.0f + d->brightness) : (1.0f - d->brightness); #ifdef _OPENMP #pragma omp parallel for default(none) shared(d) schedule(static) #endif for(int k = 0; k < 0x10000; k++) { d->ltable[k] = 100.0f * powf((float)k / 0x10000, gamma); } // now the extrapolation stuff for the brightness curve: const float xl[4] = { 0.7f, 0.8f, 0.9f, 1.0f }; const float yl[4] = { d->ltable[CLAMP((int)(xl[0] * 0x10000ul), 0, 0xffff)], d->ltable[CLAMP((int)(xl[1] * 0x10000ul), 0, 0xffff)], d->ltable[CLAMP((int)(xl[2] * 0x10000ul), 0, 0xffff)], d->ltable[CLAMP((int)(xl[3] * 0x10000ul), 0, 0xffff)] }; dt_iop_estimate_exp(xl, yl, 4, d->lunbounded_coeffs); } void init_pipe(struct dt_iop_module_t *self, dt_dev_pixelpipe_t *pipe, dt_dev_pixelpipe_iop_t *piece) { dt_iop_colisa_data_t *d = (dt_iop_colisa_data_t *)calloc(1, sizeof(dt_iop_colisa_data_t)); piece->data = (void *)d; self->commit_params(self, self->default_params, pipe, piece); for(int k = 0; k < 0x10000; k++) d->ctable[k] = d->ltable[k] = 100.0f * k / 0x10000; // identity } void cleanup_pipe(struct dt_iop_module_t *self, dt_dev_pixelpipe_t *pipe, dt_dev_pixelpipe_iop_t *piece) { free(piece->data); piece->data = NULL; } void gui_update(struct dt_iop_module_t *self) { dt_iop_module_t *module = (dt_iop_module_t *)self; dt_iop_colisa_gui_data_t *g = (dt_iop_colisa_gui_data_t *)self->gui_data; dt_iop_colisa_params_t *p = (dt_iop_colisa_params_t *)module->params; dt_bauhaus_slider_set(g->contrast, p->contrast); dt_bauhaus_slider_set(g->brightness, p->brightness); dt_bauhaus_slider_set(g->saturation, p->saturation); } void init(dt_iop_module_t *module) { module->params = calloc(1, sizeof(dt_iop_colisa_params_t)); module->default_params = calloc(1, sizeof(dt_iop_colisa_params_t)); module->default_enabled = 0; module->priority = 647; // module order created by iop_dependencies.py, do not edit! module->params_size = sizeof(dt_iop_colisa_params_t); module->gui_data = NULL; dt_iop_colisa_params_t tmp = (dt_iop_colisa_params_t){ 0, 0, 0 }; memcpy(module->params, &tmp, sizeof(dt_iop_colisa_params_t)); memcpy(module->default_params, &tmp, sizeof(dt_iop_colisa_params_t)); } void init_global(dt_iop_module_so_t *module) { const int program = 2; // basic.cl, from programs.conf dt_iop_colisa_global_data_t *gd = (dt_iop_colisa_global_data_t *)malloc(sizeof(dt_iop_colisa_global_data_t)); module->data = gd; gd->kernel_colisa = dt_opencl_create_kernel(program, "colisa"); } void cleanup(dt_iop_module_t *module) { free(module->params); module->params = NULL; } void cleanup_global(dt_iop_module_so_t *module) { dt_iop_colisa_global_data_t *gd = (dt_iop_colisa_global_data_t *)module->data; dt_opencl_free_kernel(gd->kernel_colisa); free(module->data); module->data = NULL; } void gui_init(struct dt_iop_module_t *self) { self->gui_data = malloc(sizeof(dt_iop_colisa_gui_data_t)); dt_iop_colisa_gui_data_t *g = (dt_iop_colisa_gui_data_t *)self->gui_data; dt_iop_colisa_params_t *p = (dt_iop_colisa_params_t *)self->params; self->widget = gtk_box_new(GTK_ORIENTATION_VERTICAL, DT_BAUHAUS_SPACE); g->contrast = dt_bauhaus_slider_new_with_range(self, -1.0, 1.0, 0.01, p->contrast, 2); g->brightness = dt_bauhaus_slider_new_with_range(self, -1.0, 1.0, 0.01, p->brightness, 2); g->saturation = dt_bauhaus_slider_new_with_range(self, -1.0, 1.0, 0.01, p->saturation, 2); dt_bauhaus_widget_set_label(g->contrast, NULL, _("contrast")); dt_bauhaus_widget_set_label(g->brightness, NULL, _("brightness")); dt_bauhaus_widget_set_label(g->saturation, NULL, _("saturation")); gtk_box_pack_start(GTK_BOX(self->widget), g->contrast, TRUE, TRUE, 0); gtk_box_pack_start(GTK_BOX(self->widget), g->brightness, TRUE, TRUE, 0); gtk_box_pack_start(GTK_BOX(self->widget), g->saturation, TRUE, TRUE, 0); gtk_widget_set_tooltip_text(g->contrast, _("contrast adjustment")); gtk_widget_set_tooltip_text(g->brightness, _("brightness adjustment")); gtk_widget_set_tooltip_text(g->saturation, _("color saturation adjustment")); g_signal_connect(G_OBJECT(g->contrast), "value-changed", G_CALLBACK(contrast_callback), self); g_signal_connect(G_OBJECT(g->brightness), "value-changed", G_CALLBACK(brightness_callback), self); g_signal_connect(G_OBJECT(g->saturation), "value-changed", G_CALLBACK(saturation_callback), self); } void gui_cleanup(struct dt_iop_module_t *self) { free(self->gui_data); self->gui_data = NULL; } // modelines: These editor modelines have been set for all relevant files by tools/update_modelines.sh // vim: shiftwidth=2 expandtab tabstop=2 cindent // kate: tab-indents: off; indent-width 2; replace-tabs on; indent-mode cstyle; remove-trailing-spaces modified;