4 * Copyright (C) 2008 Adam Williams <broadcast at earthling dot net>
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software
18 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
22 #include "bcdisplayinfo.h"
25 #include "bcsignals.h"
30 #include "unsharpwindow.h"
36 REGISTER_PLUGIN(UnsharpMain)
40 UnsharpConfig::UnsharpConfig()
47 int UnsharpConfig::equivalent(UnsharpConfig &that)
49 return EQUIV(radius, that.radius) &&
50 EQUIV(amount, that.amount) &&
51 threshold == that.threshold;
54 void UnsharpConfig::copy_from(UnsharpConfig &that)
58 threshold = that.threshold;
61 void UnsharpConfig::interpolate(UnsharpConfig &prev,
65 int64_t current_frame)
67 double next_scale = (double)(current_frame - prev_frame) / (next_frame - prev_frame);
68 double prev_scale = (double)(next_frame - current_frame) / (next_frame - prev_frame);
69 this->radius = prev.radius * prev_scale + next.radius * next_scale;
70 this->amount = prev.amount * prev_scale + next.amount * next_scale;
71 this->threshold = (int)(prev.threshold * prev_scale + next.threshold * next_scale);
92 UnsharpMain::UnsharpMain(PluginServer *server)
93 : PluginVClient(server)
99 UnsharpMain::~UnsharpMain()
105 const char* UnsharpMain::plugin_title() { return _("Unsharp"); }
106 int UnsharpMain::is_realtime() { return 1; }
109 NEW_WINDOW_MACRO(UnsharpMain, UnsharpWindow)
111 LOAD_CONFIGURATION_MACRO(UnsharpMain, UnsharpConfig)
115 void UnsharpMain::update_gui()
119 if(load_configuration())
121 thread->window->lock_window("UnsharpMain::update_gui");
122 ((UnsharpWindow*)thread->window)->update();
123 thread->window->unlock_window();
131 void UnsharpMain::save_data(KeyFrame *keyframe)
135 // cause data to be stored directly in text
136 output.set_shared_output(keyframe->get_data(), MESSAGESIZE);
137 output.tag.set_title("UNSHARP");
139 output.tag.set_property("RADIUS", config.radius);
140 output.tag.set_property("AMOUNT", config.amount);
141 output.tag.set_property("THRESHOLD", config.threshold);
143 output.tag.set_title("/UNSHARP");
145 output.append_newline();
146 output.terminate_string();
149 void UnsharpMain::read_data(KeyFrame *keyframe)
153 input.set_shared_input(keyframe->get_data(), strlen(keyframe->get_data()));
159 result = input.read_tag();
163 if(input.tag.title_is("UNSHARP"))
165 config.radius = input.tag.get_property("RADIUS", config.radius);
166 config.amount = input.tag.get_property("AMOUNT", config.amount);
167 config.threshold = input.tag.get_property("THRESHOLD", config.threshold);
179 int UnsharpMain::process_buffer(VFrame *frame,
180 int64_t start_position,
183 /*int need_reconfigure =*/ load_configuration();
185 if(!engine) engine = new UnsharpEngine(this,
186 get_project_smp() + 1, get_project_smp() + 1);
187 read_frame(frame, 0, get_source_position(), get_framerate(), 0);
188 engine->do_unsharp(frame);
203 UnsharpPackage::UnsharpPackage()
213 UnsharpUnit::UnsharpUnit(UnsharpEngine *server,
217 this->plugin = plugin;
218 this->server = server;
222 UnsharpUnit::~UnsharpUnit()
228 // Derived from the Gimp.
229 // In the original file it says
231 // Copyright (C) 1999 Winston Chang
232 // <winstonc@cs.wisc.edu>
233 // <winston@stdout.org>
235 // Adapted for Cinelerra by Heroine Virtual Ltd.
237 static int calculate_convolution_matrix(double radius, double **cmatrix)
239 radius = fabs(radius) + 1.0;
240 double std_dev = radius;
241 radius = std_dev * 2;
242 int matrix_length = (int)(2 * ceil(radius - 0.5) + 1);
243 matrix_length = MAX(1, matrix_length);
244 // int matrix_midpoint = matrix_length / 2 + 1;
245 (*cmatrix) = new double[matrix_length];
247 // Top right of matrix
248 for(int i = matrix_length / 2 + 1; i < matrix_length; i++)
250 double base_x = i - floor(matrix_length / 2) - 0.5;
252 for(int j = 1; j <= 50; j++)
254 if(base_x + 0.02 * j <= radius)
256 sum += exp(-(base_x + 0.02 * j) *
257 (base_x + 0.02 * j) /
258 (2 * std_dev * std_dev));
261 (*cmatrix)[i] = sum / 50;
264 // Top left of matrix
265 for(int i = 0; i < matrix_length / 2; i++)
267 (*cmatrix)[i] = (*cmatrix)[matrix_length - 1 - i];
272 for(int j = 0; j <= 50; j++)
274 sum += exp(-(0.5 + 0.02 * j) *
276 (2 * std_dev * std_dev));
278 (*cmatrix)[matrix_length / 2] = sum / 51;
282 for(int i = 0; i < matrix_length; i++)
283 sum += (*cmatrix)[i];
284 for(int i = 0; i < matrix_length; i++)
285 (*cmatrix)[i] = (*cmatrix)[i] / sum;
287 return matrix_length;
290 static double get_convolution(double *cmatrix,
294 return cmatrix[index] * input;
297 static void blur_pixels(double *cmatrix,
304 if(cmatrix_length > pixels)
306 for(int pixel = 0; pixel < pixels; pixel++)
309 for(int j = 0; j < pixels; j++)
311 if((j + cmatrix_length / 2 - pixel >= 0) &&
312 (j + cmatrix_length / 2 - pixel < cmatrix_length))
314 scale += cmatrix[j + cmatrix_length / 2 - pixel];
318 for(int i = 0; i < components; i++)
321 for(int j = 0; j < pixels; j++)
323 if((j >= pixel - cmatrix_length / 2) &&
324 (j <= pixel + cmatrix_length / 2))
326 sum += input[j * components + i] * cmatrix[i];
329 output[pixel * components + i] = sum / scale;
335 int cmatrix_middle = cmatrix_length / 2;
337 for(pixel = 0; pixel < cmatrix_middle; pixel++)
340 for(int j = cmatrix_middle - pixel; j < cmatrix_length;j++)
345 for(int i = 0; i < components; i++)
348 for(int j = cmatrix_middle - pixel; j < cmatrix_length; j++)
350 sum += input[(pixel + j - cmatrix_middle) * components + i] *
353 output[pixel * components + i] = sum / scale;
357 float *output_ptr = output + pixel * components;
358 for( ; pixel < pixels - cmatrix_middle; pixel++)
360 float *input_ptr = input + (pixel - cmatrix_middle) * components;
361 for(int i = 0; i < components; i++)
364 float *input_ptr2 = input_ptr;
365 for(int j = cmatrix_length; j > 0; j--)
367 sum += get_convolution(cmatrix,
370 input_ptr2 += components;
377 for( ; pixel < pixels; pixel++)
380 for(int j = 0; j < pixels - pixel + cmatrix_middle; j++)
385 for(int i = 0; i < components; i++)
388 for(int j = 0; j < pixels - pixel + cmatrix_middle; j++)
390 sum += input[(pixel + j - cmatrix_middle) * components + i] *
393 output[pixel * components + i] = sum / scale;
399 #define GET_ROW(type, components) \
401 type *in_row = (type*)src->get_rows()[row]; \
402 int pixels = src->get_w() * components; \
403 for(int i = 0; i < pixels; i++) \
405 dst[i] = in_row[i]; \
409 static void get_row(float *dst, VFrame *src, int row)
411 switch(src->get_color_model())
415 GET_ROW(unsigned char, 3);
422 GET_ROW(unsigned char, 4);
428 GET_ROW(uint16_t, 3);
430 case BC_YUVA16161616:
431 GET_ROW(uint16_t, 4);
436 static void get_column(float *dst, VFrame *src, int column)
438 int components = BC_CModels::components(src->get_color_model());
439 for(int i = 0; i < src->get_h(); i++)
441 float *input_pixel = (float*)src->get_rows()[i] + column * components;
442 memcpy(dst, input_pixel, sizeof(float) * components);
447 static void put_column(float *src, VFrame *dst, int column)
449 int components = BC_CModels::components(dst->get_color_model());
450 for(int i = 0; i < dst->get_h(); i++)
452 float *output_pixel = (float*)dst->get_rows()[i] + column * components;
453 memcpy(output_pixel, src, sizeof(float) * components);
458 void UnsharpUnit::process_package(LoadPackage *package)
460 UnsharpPackage *pkg = (UnsharpPackage*)package;
461 // int w = server->src->get_w();
462 // int h = server->src->get_h();
463 int color_model = server->src->get_color_model();
464 int components = BC_CModels::components(color_model);
466 int cmatrix_length = 0;
467 int padded_y1 = pkg->y1;
468 int padded_y2 = pkg->y2;
470 cmatrix_length = calculate_convolution_matrix(
471 plugin->config.radius,
475 if(padded_y2 < server->src->get_h())
477 padded_y2 += cmatrix_length / 2;
478 padded_y2 = MIN(server->src->get_h(), padded_y2);
482 padded_y1 -= cmatrix_length / 2;
483 padded_y1 = MAX(0, padded_y1);
486 int padded_rows = padded_y2 - padded_y1;
488 if(!temp || temp->get_h() != padded_rows)
497 temp->set_use_shm(0);
503 server->src->get_w(),
505 components == 3 ? BC_RGB_FLOAT : BC_RGBA_FLOAT,
509 float *temp_in = new float[MAX(temp->get_w(), padded_rows) * components];
510 float *temp_out = new float[MAX(temp->get_w(), padded_rows) * components];
513 for(int i = padded_y1; i < padded_y2; i++)
515 get_row(temp_in, server->src, i);
522 // printf("UnsharpUnit::process_package %d %p %p %p %d %d\n",
525 // temp->get_rows()[0],
528 // temp->get_bytes_per_line());
529 memcpy(temp->get_rows()[i - padded_y1],
531 temp->get_bytes_per_line());
534 //Now we're 100% floating point. Blur the columns
535 for(int i = 0; i < temp->get_w(); i++)
537 get_column(temp_in, temp, i);
544 put_column(temp_out, temp, i);
548 //printf("%f %f %d\n", plugin->config.radius,plugin->config.amount, plugin->config.threshold);
551 #define UNSHARPEN(type, components, max) \
553 float threshold = (float)plugin->config.threshold * max / 0xff; \
554 float amount = plugin->config.amount; \
556 for(int i = pkg->y1; i < pkg->y2; i++) \
558 float *blurry_row = (float*)temp->get_rows()[i - padded_y1]; \
559 type *orig_row = (type*)server->src->get_rows()[i]; \
560 for(int j = 0; j < server->src->get_w(); j++) \
562 for(int k = 0; k < components; k++) \
564 float diff = *orig_row - *blurry_row; \
565 if(fabsf(2 * diff) < threshold) \
567 float value = *orig_row + amount * diff; \
568 if(sizeof(type) == 4) \
569 *orig_row = (type)value; \
571 *orig_row = (type)CLIP(value, 0, max); \
579 // Apply unsharpening
584 UNSHARPEN(unsigned char, 3, 0xff);
588 UNSHARPEN(unsigned char, 4, 0xff);
591 UNSHARPEN(float, 3, 1.0);
594 UNSHARPEN(float, 4, 1.0);
597 UNSHARPEN(uint16_t, 3, 0xffff);
599 case BC_YUVA16161616:
600 UNSHARPEN(uint16_t, 4, 0xffff);
617 UnsharpEngine::UnsharpEngine(UnsharpMain *plugin,
622 total_clients, total_packages
625 this->plugin = plugin;
628 UnsharpEngine::~UnsharpEngine()
633 void UnsharpEngine::init_packages()
635 for(int i = 0; i < get_total_packages(); i++)
637 UnsharpPackage *pkg = (UnsharpPackage*)get_package(i);
638 pkg->y1 = src->get_h() * i / get_total_packages();
639 pkg->y2 = src->get_h() * (i + 1) / get_total_packages();
643 LoadClient* UnsharpEngine::new_client()
645 return new UnsharpUnit(this, plugin);
648 LoadPackage* UnsharpEngine::new_package()
650 return new UnsharpPackage;
654 void UnsharpEngine::do_unsharp(VFrame *src)