43#include "MagickCore/studio.h"
44#include "MagickCore/accelerate-private.h"
45#include "MagickCore/blob.h"
46#include "MagickCore/cache-view.h"
47#include "MagickCore/color.h"
48#include "MagickCore/color-private.h"
49#include "MagickCore/colorspace.h"
50#include "MagickCore/constitute.h"
51#include "MagickCore/decorate.h"
52#include "MagickCore/distort.h"
53#include "MagickCore/draw.h"
54#include "MagickCore/enhance.h"
55#include "MagickCore/exception.h"
56#include "MagickCore/exception-private.h"
57#include "MagickCore/effect.h"
58#include "MagickCore/fx.h"
59#include "MagickCore/gem.h"
60#include "MagickCore/gem-private.h"
61#include "MagickCore/geometry.h"
62#include "MagickCore/image-private.h"
63#include "MagickCore/list.h"
64#include "MagickCore/log.h"
65#include "MagickCore/matrix.h"
66#include "MagickCore/memory_.h"
67#include "MagickCore/memory-private.h"
68#include "MagickCore/monitor.h"
69#include "MagickCore/monitor-private.h"
70#include "MagickCore/montage.h"
71#include "MagickCore/morphology.h"
72#include "MagickCore/morphology-private.h"
73#include "MagickCore/paint.h"
74#include "MagickCore/pixel-accessor.h"
75#include "MagickCore/property.h"
76#include "MagickCore/quantize.h"
77#include "MagickCore/quantum.h"
78#include "MagickCore/quantum-private.h"
79#include "MagickCore/random_.h"
80#include "MagickCore/random-private.h"
81#include "MagickCore/resample.h"
82#include "MagickCore/resample-private.h"
83#include "MagickCore/resize.h"
84#include "MagickCore/resource_.h"
85#include "MagickCore/segment.h"
86#include "MagickCore/shear.h"
87#include "MagickCore/signature-private.h"
88#include "MagickCore/statistic.h"
89#include "MagickCore/string_.h"
90#include "MagickCore/thread-private.h"
91#include "MagickCore/transform.h"
92#include "MagickCore/threshold.h"
93#include "MagickCore/utility-private.h"
129MagickExport Image *AdaptiveBlurImage(
const Image *image,
const double radius,
130 const double sigma,ExceptionInfo *exception)
132#define AdaptiveBlurImageTag "Convolve/Image"
133#define MagickSigma (fabs(sigma) < MagickEpsilon ? MagickEpsilon : sigma)
162 assert(image != (
const Image *) NULL);
163 assert(image->signature == MagickCoreSignature);
164 assert(exception != (ExceptionInfo *) NULL);
165 assert(exception->signature == MagickCoreSignature);
166 if (IsEventLogging() != MagickFalse)
167 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
168 blur_image=CloneImage(image,0,0,MagickTrue,exception);
169 if (blur_image == (Image *) NULL)
170 return((Image *) NULL);
171 if (fabs(sigma) < MagickEpsilon)
173 if (SetImageStorageClass(blur_image,DirectClass,exception) == MagickFalse)
175 blur_image=DestroyImage(blur_image);
176 return((Image *) NULL);
181 edge_image=EdgeImage(image,radius,exception);
182 if (edge_image == (Image *) NULL)
184 blur_image=DestroyImage(blur_image);
185 return((Image *) NULL);
187 (void) AutoLevelImage(edge_image,exception);
188 gaussian_image=BlurImage(edge_image,radius,sigma,exception);
189 if (gaussian_image != (Image *) NULL)
191 edge_image=DestroyImage(edge_image);
192 edge_image=gaussian_image;
194 (void) AutoLevelImage(edge_image,exception);
198 width=GetOptimalKernelWidth2D(radius,sigma);
199 kernel=(
double **) MagickAssumeAligned(AcquireAlignedMemory((
size_t) width,
201 if (kernel == (
double **) NULL)
203 edge_image=DestroyImage(edge_image);
204 blur_image=DestroyImage(blur_image);
205 ThrowImageException(ResourceLimitError,
"MemoryAllocationFailed");
207 (void) memset(kernel,0,(
size_t) width*
sizeof(*kernel));
208 for (w=0; w < (ssize_t) width; w+=2)
216 kernel[w]=(
double *) MagickAssumeAligned(AcquireAlignedMemory(
217 (width-(
size_t) w),(width-(
size_t) w)*
sizeof(**kernel)));
218 if (kernel[w] == (
double *) NULL)
221 j=((ssize_t) width-w-1)/2;
223 for (v=(-j); v <= j; v++)
225 for (u=(-j); u <= j; u++)
227 kernel[w][k]=(double) (exp(-((
double) u*u+v*v)/(2.0*MagickSigma*
228 MagickSigma))/(2.0*MagickPI*MagickSigma*MagickSigma));
229 normalize+=kernel[w][k];
233 kernel[w][(k-1)/2]+=(
double) (1.0-normalize);
234 if (sigma < MagickEpsilon)
235 kernel[w][(k-1)/2]=1.0;
237 if (w < (ssize_t) width)
239 for (w-=2; w >= 0; w-=2)
240 kernel[w]=(
double *) RelinquishAlignedMemory(kernel[w]);
241 kernel=(
double **) RelinquishAlignedMemory(kernel);
242 edge_image=DestroyImage(edge_image);
243 blur_image=DestroyImage(blur_image);
244 ThrowImageException(ResourceLimitError,
"MemoryAllocationFailed");
251 image_view=AcquireVirtualCacheView(image,exception);
252 edge_view=AcquireVirtualCacheView(edge_image,exception);
253 blur_view=AcquireAuthenticCacheView(blur_image,exception);
254#if defined(MAGICKCORE_OPENMP_SUPPORT)
255 #pragma omp parallel for schedule(static) shared(progress,status) \
256 magick_number_threads(image,blur_image,blur_image->rows,1)
258 for (y=0; y < (ssize_t) blur_image->rows; y++)
269 if (status == MagickFalse)
271 r=GetCacheViewVirtualPixels(edge_view,0,y,edge_image->columns,1,exception);
272 q=QueueCacheViewAuthenticPixels(blur_view,0,y,blur_image->columns,1,
274 if ((r == (
const Quantum *) NULL) || (q == (Quantum *) NULL))
279 for (x=0; x < (ssize_t) blur_image->columns; x++)
291 j=CastDoubleToSsizeT(ceil((
double) width*(1.0-QuantumScale*
292 GetPixelIntensity(edge_image,r))-0.5));
296 if (j > (ssize_t) width)
300 p=GetCacheViewVirtualPixels(image_view,x-((ssize_t) width-j)/2L,y-
301 ((ssize_t) width-j)/2L,width-(
size_t) j,width-(
size_t) j,exception);
302 if (p == (
const Quantum *) NULL)
304 center=(ssize_t) (GetPixelChannels(image)*(width-(
size_t) j)*
305 ((width-(
size_t) j)/2L)+GetPixelChannels(image)*((width-(
size_t) j)/2));
306 for (i=0; i < (ssize_t) GetPixelChannels(blur_image); i++)
312 *magick_restrict pixels;
330 channel=GetPixelChannelChannel(image,i);
331 traits=GetPixelChannelTraits(image,channel);
332 blur_traits=GetPixelChannelTraits(blur_image,channel);
333 if ((traits == UndefinedPixelTrait) ||
334 (blur_traits == UndefinedPixelTrait))
336 if ((blur_traits & CopyPixelTrait) != 0)
338 SetPixelChannel(blur_image,channel,p[center+i],q);
345 if ((blur_traits & BlendPixelTrait) == 0)
350 for (v=0; v < ((ssize_t) width-j); v++)
352 for (u=0; u < ((ssize_t) width-j); u++)
354 pixel+=(*k)*(double) pixels[i];
357 pixels+=(ptrdiff_t) GetPixelChannels(image);
360 gamma=MagickSafeReciprocal(gamma);
361 SetPixelChannel(blur_image,channel,ClampToQuantum(gamma*pixel),q);
367 for (v=0; v < ((ssize_t) width-j); v++)
369 for (u=0; u < ((ssize_t) width-j); u++)
371 alpha=(double) (QuantumScale*(
double) GetPixelAlpha(image,pixels));
372 pixel+=(*k)*alpha*(double) pixels[i];
375 pixels+=(ptrdiff_t) GetPixelChannels(image);
378 gamma=MagickSafeReciprocal(gamma);
379 SetPixelChannel(blur_image,channel,ClampToQuantum(gamma*pixel),q);
381 q+=(ptrdiff_t) GetPixelChannels(blur_image);
382 r+=(ptrdiff_t) GetPixelChannels(edge_image);
384 if (SyncCacheViewAuthenticPixels(blur_view,exception) == MagickFalse)
386 if (image->progress_monitor != (MagickProgressMonitor) NULL)
391#if defined(MAGICKCORE_OPENMP_SUPPORT)
395 proceed=SetImageProgress(image,AdaptiveBlurImageTag,progress,
397 if (proceed == MagickFalse)
401 blur_image->type=image->type;
402 blur_view=DestroyCacheView(blur_view);
403 edge_view=DestroyCacheView(edge_view);
404 image_view=DestroyCacheView(image_view);
405 edge_image=DestroyImage(edge_image);
406 for (w=0; w < (ssize_t) width; w+=2)
407 kernel[w]=(
double *) RelinquishAlignedMemory(kernel[w]);
408 kernel=(
double **) RelinquishAlignedMemory(kernel);
409 if (status == MagickFalse)
410 blur_image=DestroyImage(blur_image);
448MagickExport Image *AdaptiveSharpenImage(
const Image *image,
const double radius,
449 const double sigma,ExceptionInfo *exception)
451#define AdaptiveSharpenImageTag "Convolve/Image"
452#define MagickSigma (fabs(sigma) < MagickEpsilon ? MagickEpsilon : sigma)
481 assert(image != (
const Image *) NULL);
482 assert(image->signature == MagickCoreSignature);
483 assert(exception != (ExceptionInfo *) NULL);
484 assert(exception->signature == MagickCoreSignature);
485 if (IsEventLogging() != MagickFalse)
486 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
487 sharp_image=CloneImage(image,0,0,MagickTrue,exception);
488 if (sharp_image == (Image *) NULL)
489 return((Image *) NULL);
490 if (fabs(sigma) < MagickEpsilon)
492 if (SetImageStorageClass(sharp_image,DirectClass,exception) == MagickFalse)
494 sharp_image=DestroyImage(sharp_image);
495 return((Image *) NULL);
500 edge_image=EdgeImage(image,radius,exception);
501 if (edge_image == (Image *) NULL)
503 sharp_image=DestroyImage(sharp_image);
504 return((Image *) NULL);
506 (void) AutoLevelImage(edge_image,exception);
507 gaussian_image=BlurImage(edge_image,radius,sigma,exception);
508 if (gaussian_image != (Image *) NULL)
510 edge_image=DestroyImage(edge_image);
511 edge_image=gaussian_image;
513 (void) AutoLevelImage(edge_image,exception);
517 width=GetOptimalKernelWidth2D(radius,sigma);
518 kernel=(
double **) MagickAssumeAligned(AcquireAlignedMemory((
size_t)
519 width,
sizeof(*kernel)));
520 if (kernel == (
double **) NULL)
522 edge_image=DestroyImage(edge_image);
523 sharp_image=DestroyImage(sharp_image);
524 ThrowImageException(ResourceLimitError,
"MemoryAllocationFailed");
526 (void) memset(kernel,0,(
size_t) width*
sizeof(*kernel));
527 for (w=0; w < (ssize_t) width; w+=2)
535 kernel[w]=(
double *) MagickAssumeAligned(AcquireAlignedMemory((
size_t)
536 (width-(
size_t) w),(width-(
size_t) w)*
sizeof(**kernel)));
537 if (kernel[w] == (
double *) NULL)
540 j=((ssize_t) width-w-1)/2;
542 for (v=(-j); v <= j; v++)
544 for (u=(-j); u <= j; u++)
546 kernel[w][k]=(double) (-exp(-((
double) u*u+v*v)/(2.0*MagickSigma*
547 MagickSigma))/(2.0*MagickPI*MagickSigma*MagickSigma));
548 normalize+=kernel[w][k];
552 kernel[w][(k-1)/2]=(
double) ((-2.0)*normalize);
553 if (sigma < MagickEpsilon)
554 kernel[w][(k-1)/2]=1.0;
556 if (w < (ssize_t) width)
558 for (w-=2; w >= 0; w-=2)
559 kernel[w]=(
double *) RelinquishAlignedMemory(kernel[w]);
560 kernel=(
double **) RelinquishAlignedMemory(kernel);
561 edge_image=DestroyImage(edge_image);
562 sharp_image=DestroyImage(sharp_image);
563 ThrowImageException(ResourceLimitError,
"MemoryAllocationFailed");
570 image_view=AcquireVirtualCacheView(image,exception);
571 edge_view=AcquireVirtualCacheView(edge_image,exception);
572 sharp_view=AcquireAuthenticCacheView(sharp_image,exception);
573#if defined(MAGICKCORE_OPENMP_SUPPORT)
574 #pragma omp parallel for schedule(static) shared(progress,status) \
575 magick_number_threads(image,sharp_image,sharp_image->rows,1)
577 for (y=0; y < (ssize_t) sharp_image->rows; y++)
588 if (status == MagickFalse)
590 r=GetCacheViewVirtualPixels(edge_view,0,y,edge_image->columns,1,exception);
591 q=QueueCacheViewAuthenticPixels(sharp_view,0,y,sharp_image->columns,1,
593 if ((r == (
const Quantum *) NULL) || (q == (Quantum *) NULL))
598 for (x=0; x < (ssize_t) sharp_image->columns; x++)
610 j=CastDoubleToSsizeT(ceil((
double) width*(1.0-QuantumScale*
611 GetPixelIntensity(edge_image,r))-0.5));
615 if (j > (ssize_t) width)
619 p=GetCacheViewVirtualPixels(image_view,x-(((ssize_t) width-j)/2L),y-
620 (((ssize_t) width-j)/2L),width-(
size_t) j,width-(
size_t) j,exception);
621 if (p == (
const Quantum *) NULL)
623 center=(ssize_t) (GetPixelChannels(image)*(width-(
size_t) j)*
624 ((width-(
size_t) j)/2L)+GetPixelChannels(image)*((width-(
size_t) j)/2));
625 for (i=0; i < (ssize_t) GetPixelChannels(sharp_image); i++)
631 *magick_restrict pixels;
649 channel=GetPixelChannelChannel(image,i);
650 traits=GetPixelChannelTraits(image,channel);
651 sharp_traits=GetPixelChannelTraits(sharp_image,channel);
652 if ((traits == UndefinedPixelTrait) ||
653 (sharp_traits == UndefinedPixelTrait))
655 if ((sharp_traits & CopyPixelTrait) != 0)
657 SetPixelChannel(sharp_image,channel,p[center+i],q);
664 if ((sharp_traits & BlendPixelTrait) == 0)
669 for (v=0; v < ((ssize_t) width-j); v++)
671 for (u=0; u < ((ssize_t) width-j); u++)
673 pixel+=(*k)*(double) pixels[i];
676 pixels+=(ptrdiff_t) GetPixelChannels(image);
679 gamma=MagickSafeReciprocal(gamma);
680 SetPixelChannel(sharp_image,channel,ClampToQuantum(gamma*pixel),q);
686 for (v=0; v < ((ssize_t) width-j); v++)
688 for (u=0; u < ((ssize_t) width-j); u++)
690 alpha=(double) (QuantumScale*(
double) GetPixelAlpha(image,pixels));
691 pixel+=(*k)*alpha*(double) pixels[i];
694 pixels+=(ptrdiff_t) GetPixelChannels(image);
697 gamma=MagickSafeReciprocal(gamma);
698 SetPixelChannel(sharp_image,channel,ClampToQuantum(gamma*pixel),q);
700 q+=(ptrdiff_t) GetPixelChannels(sharp_image);
701 r+=(ptrdiff_t) GetPixelChannels(edge_image);
703 if (SyncCacheViewAuthenticPixels(sharp_view,exception) == MagickFalse)
705 if (image->progress_monitor != (MagickProgressMonitor) NULL)
710#if defined(MAGICKCORE_OPENMP_SUPPORT)
714 proceed=SetImageProgress(image,AdaptiveSharpenImageTag,progress,
716 if (proceed == MagickFalse)
720 sharp_image->type=image->type;
721 sharp_view=DestroyCacheView(sharp_view);
722 edge_view=DestroyCacheView(edge_view);
723 image_view=DestroyCacheView(image_view);
724 edge_image=DestroyImage(edge_image);
725 for (w=0; w < (ssize_t) width; w+=2)
726 kernel[w]=(
double *) RelinquishAlignedMemory(kernel[w]);
727 kernel=(
double **) RelinquishAlignedMemory(kernel);
728 if (status == MagickFalse)
729 sharp_image=DestroyImage(sharp_image);
766MagickExport Image *BlurImage(
const Image *image,
const double radius,
767 const double sigma,ExceptionInfo *exception)
770 geometry[MagickPathExtent];
778 assert(image != (
const Image *) NULL);
779 assert(image->signature == MagickCoreSignature);
780 assert(exception != (ExceptionInfo *) NULL);
781 assert(exception->signature == MagickCoreSignature);
782 if (IsEventLogging() != MagickFalse)
783 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
784#if defined(MAGICKCORE_OPENCL_SUPPORT)
785 blur_image=AccelerateBlurImage(image,radius,sigma,exception);
786 if (blur_image != (Image *) NULL)
789 (void) FormatLocaleString(geometry,MagickPathExtent,
790 "blur:%.20gx%.20g;blur:%.20gx%.20g+90",radius,sigma,radius,sigma);
791 kernel_info=AcquireKernelInfo(geometry,exception);
792 if (kernel_info == (KernelInfo *) NULL)
793 ThrowImageException(ResourceLimitError,
"MemoryAllocationFailed");
794 blur_image=ConvolveImage(image,kernel_info,exception);
795 kernel_info=DestroyKernelInfo(kernel_info);
847static inline double BlurDistance(
const ssize_t x,
const ssize_t y,
848 const ssize_t u,
const ssize_t v)
850 return(sqrt(((
double) x-u)*((
double) x-u)+((
double) y-v)*((
double) y-v)));
853static inline double BlurGaussian(
const double x,
const double sigma)
855 return(exp(-((
double) x*x)*MagickSafeReciprocal(2.0*sigma*sigma))*
856 MagickSafeReciprocal(Magick2PI*sigma*sigma));
859static double **DestroyBilateralTLS(
const size_t number_threads,
865 assert(weights != (
double **) NULL);
866 for (i=0; i <= (ssize_t) number_threads; i++)
867 if (weights[i] != (
double *) NULL)
868 weights[i]=(
double *) RelinquishMagickMemory(weights[i]);
869 weights=(
double **) RelinquishMagickMemory(weights);
873static double **AcquireBilateralTLS(
const size_t number_threads,
874 const size_t width,
const size_t height)
885 if (HeapOverflowSanityCheckGetSize(height,
sizeof(**weights),&count) != MagickFalse)
886 return((
double **) NULL);
887 weights=(
double **) AcquireQuantumMemory(number_threads+1,
sizeof(*weights));
888 if (weights == (
double **) NULL)
889 return((
double **) NULL);
890 (void) memset(weights,0,(number_threads+1)*
sizeof(*weights));
891 for (i=0; i <= (ssize_t) number_threads; i++)
893 weights[i]=(
double *) AcquireQuantumMemory(width,count);
894 if (weights[i] == (
double *) NULL)
895 return(DestroyBilateralTLS(number_threads,weights));
900MagickExport Image *BilateralBlurImage(
const Image *image,
const size_t width,
901 const size_t height,
const double intensity_sigma,
const double spatial_sigma,
902 ExceptionInfo *exception)
904#define MaxIntensity (255)
905#define BilateralBlurImageTag "Blur/Image"
912 intensity_gaussian[2*(MaxIntensity+1)],
935 assert(image != (
const Image *) NULL);
936 assert(image->signature == MagickCoreSignature);
937 assert(exception != (ExceptionInfo *) NULL);
938 assert(exception->signature == MagickCoreSignature);
939 if (IsEventLogging() != MagickFalse)
940 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
941 blur_image=CloneImage(image,0,0,MagickTrue,exception);
942 if (blur_image == (Image *) NULL)
943 return((Image *) NULL);
944 if (SetImageStorageClass(blur_image,DirectClass,exception) == MagickFalse)
946 blur_image=DestroyImage(blur_image);
947 return((Image *) NULL);
949 number_threads=(size_t) GetMagickResourceLimit(ThreadResource);
950 weights=AcquireBilateralTLS(number_threads,MagickMax(width,1),
951 MagickMax(height,1));
952 if (weights == (
double **) NULL)
954 blur_image=DestroyImage(blur_image);
955 ThrowImageException(ResourceLimitError,
"MemoryAllocationFailed");
957 for (w=(-MaxIntensity); w <= MaxIntensity; w++)
958 intensity_gaussian[w+MaxIntensity]=BlurGaussian((
double) w,intensity_sigma);
959 spatial_gaussian=weights[number_threads];
966 mid.x=(ssize_t) (MagickMax(width,1)/2L);
967 mid.y=(ssize_t) (MagickMax(height,1)/2L);
968 for (v=0; v < (ssize_t) MagickMax(height,1); v++)
973 for (u=0; u < (ssize_t) MagickMax(width,1); u++)
974 spatial_gaussian[n++]=BlurGaussian(BlurDistance(0,0,u-mid.x,v-mid.y),
983 image_view=AcquireVirtualCacheView(image,exception);
984 blur_view=AcquireAuthenticCacheView(blur_image,exception);
985#if defined(MAGICKCORE_OPENMP_SUPPORT)
986 #pragma omp parallel for schedule(static) shared(progress,status) \
987 magick_number_threads(image,blur_image,blur_image->rows,1)
989 for (y=0; y < (ssize_t) blur_image->rows; y++)
992 id = GetOpenMPThreadId();
1000 if (status == MagickFalse)
1002 q=QueueCacheViewAuthenticPixels(blur_view,0,y,blur_image->columns,1,
1004 if (q == (Quantum *) NULL)
1009 for (x=0; x < (ssize_t) blur_image->columns; x++)
1028 p=GetCacheViewVirtualPixels(image_view,x-mid.x,y-mid.y,MagickMax(width,1),
1029 MagickMax(height,1),exception);
1030 if (p == (
const Quantum *) NULL)
1032 p+=(ptrdiff_t) (GetPixelChannels(image)*MagickMax(width,1)*(
size_t) mid.y+
1033 GetPixelChannels(image)*(
size_t) mid.x);
1035 for (v=0; v < (ssize_t) MagickMax(height,1); v++)
1037 for (u=0; u < (ssize_t) MagickMax(width,1); u++)
1042 r=p-(ssize_t) (GetPixelChannels(image)*MagickMax(width,1)*
1043 (
size_t) (mid.y-v)+GetPixelChannels(image)*(
size_t) (mid.x-u));
1044 intensity=ScaleQuantumToChar((
const Quantum) GetPixelIntensity(image,r))-
1045 (double) ScaleQuantumToChar((
const Quantum) GetPixelIntensity(image,p));
1046 if ((intensity >= -MaxIntensity) && (intensity <= MaxIntensity))
1047 weights[id][n]=intensity_gaussian[(ssize_t) intensity+MaxIntensity]*
1048 spatial_gaussian[n];
1050 weights[id][n]=BlurGaussian(intensity,intensity_sigma)*
1051 BlurGaussian(BlurDistance(x,y,x+u-mid.x,y+v-mid.y),spatial_sigma);
1055 for (i=0; i < (ssize_t) GetPixelChannels(blur_image); i++)
1064 channel=GetPixelChannelChannel(image,i);
1065 traits=GetPixelChannelTraits(image,channel);
1066 blur_traits=GetPixelChannelTraits(blur_image,channel);
1067 if ((traits == UndefinedPixelTrait) ||
1068 (blur_traits == UndefinedPixelTrait))
1070 if ((blur_traits & CopyPixelTrait) != 0)
1072 SetPixelChannel(blur_image,channel,p[i],q);
1078 if ((blur_traits & BlendPixelTrait) == 0)
1083 for (v=0; v < (ssize_t) MagickMax(height,1); v++)
1085 for (u=0; u < (ssize_t) MagickMax(width,1); u++)
1087 r=p-(ssize_t) (GetPixelChannels(image)*MagickMax(width,1)*(
size_t)
1088 (mid.y-v)+GetPixelChannels(image)*(
size_t) (mid.x-u));
1089 pixel+=weights[id][n]*(double) r[i];
1090 gamma+=weights[id][n];
1094 SetPixelChannel(blur_image,channel,ClampToQuantum(
1095 MagickSafeReciprocal(gamma)*pixel),q);
1101 for (v=0; v < (ssize_t) MagickMax(height,1); v++)
1103 for (u=0; u < (ssize_t) MagickMax(width,1); u++)
1109 r=p-(ssize_t) (GetPixelChannels(image)*MagickMax(width,1)*(
size_t) (mid.y-v)+
1110 GetPixelChannels(image)*(
size_t) (mid.x-u));
1111 alpha=(double) (QuantumScale*(
double) GetPixelAlpha(image,p));
1112 beta=(double) (QuantumScale*(
double) GetPixelAlpha(image,r));
1113 pixel+=weights[id][n]*(double) r[i];
1114 gamma+=weights[id][n]*alpha*beta;
1118 SetPixelChannel(blur_image,channel,ClampToQuantum(
1119 MagickSafeReciprocal(gamma)*pixel),q);
1121 q+=(ptrdiff_t) GetPixelChannels(blur_image);
1123 if (SyncCacheViewAuthenticPixels(blur_view,exception) == MagickFalse)
1125 if (image->progress_monitor != (MagickProgressMonitor) NULL)
1130#if defined(MAGICKCORE_OPENMP_SUPPORT)
1134 proceed=SetImageProgress(image,BilateralBlurImageTag,progress,
1136 if (proceed == MagickFalse)
1140 blur_image->type=image->type;
1141 blur_view=DestroyCacheView(blur_view);
1142 image_view=DestroyCacheView(image_view);
1143 weights=DestroyBilateralTLS(number_threads,weights);
1144 if (status == MagickFalse)
1145 blur_image=DestroyImage(blur_image);
1176MagickExport Image *ConvolveImage(
const Image *image,
1177 const KernelInfo *kernel_info,ExceptionInfo *exception)
1182 convolve_image=MorphologyImage(image,ConvolveMorphology,1,kernel_info,
1184 return(convolve_image);
1217static void Hull(
const Image *image,
const ssize_t x_offset,
1218 const ssize_t y_offset,
const size_t columns,
const size_t rows,
1219 const int polarity,Quantum *magick_restrict f,Quantum *magick_restrict g)
1230 assert(image != (
const Image *) NULL);
1231 assert(image->signature == MagickCoreSignature);
1232 assert(f != (Quantum *) NULL);
1233 assert(g != (Quantum *) NULL);
1234 assert(columns <= (
size_t) (MAGICK_SSIZE_MAX-2));
1235 if (IsEventLogging() != MagickFalse)
1236 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
1237 p=f+(ptrdiff_t) (columns+2);
1238 q=g+(ptrdiff_t) (columns+2);
1239 r=p+(ptrdiff_t) (y_offset*((ssize_t) columns+2)+x_offset);
1240#if defined(MAGICKCORE_OPENMP_SUPPORT)
1241 #pragma omp parallel for schedule(static) \
1242 magick_number_threads(image,image,rows,2)
1244 for (y=0; y < (ssize_t) rows; y++)
1253 i=(2*y+1)+y*(ssize_t) columns;
1255 for (x=0; x < (ssize_t) columns; x++)
1257 v=(MagickRealType) p[i];
1258 if ((MagickRealType) r[i] >= (v+(
double) ScaleCharToQuantum(2)))
1259 v+=(double) ScaleCharToQuantum(1);
1264 for (x=0; x < (ssize_t) columns; x++)
1266 v=(MagickRealType) p[i];
1267 if ((MagickRealType) r[i] <= (v-(
double) ScaleCharToQuantum(2)))
1268 v-=(double) ScaleCharToQuantum(1);
1273 p=f+(ptrdiff_t) (columns+2);
1274 q=g+(ptrdiff_t) (columns+2);
1275 r=q+(ptrdiff_t) (y_offset*((ssize_t) columns+2)+x_offset);
1276 s=q-(ptrdiff_t) (y_offset*((ssize_t) columns+2)+x_offset);
1277#if defined(MAGICKCORE_OPENMP_SUPPORT)
1278 #pragma omp parallel for schedule(static) \
1279 magick_number_threads(image,image,rows,2)
1281 for (y=0; y < (ssize_t) rows; y++)
1290 i=(2*y+1)+y*(ssize_t) columns;
1292 for (x=0; x < (ssize_t) columns; x++)
1294 v=(MagickRealType) q[i];
1295 if (((MagickRealType) s[i] >= (v+(
double) ScaleCharToQuantum(2))) &&
1296 ((MagickRealType) r[i] > v))
1297 v+=(double) ScaleCharToQuantum(1);
1302 for (x=0; x < (ssize_t) columns; x++)
1304 v=(MagickRealType) q[i];
1305 if (((MagickRealType) s[i] <= (v-(
double) ScaleCharToQuantum(2))) &&
1306 ((MagickRealType) r[i] < v))
1307 v-=(double) ScaleCharToQuantum(1);
1314MagickExport Image *DespeckleImage(
const Image *image,ExceptionInfo *exception)
1316#define DespeckleImageTag "Despeckle/Image"
1333 *magick_restrict buffer,
1334 *magick_restrict pixels;
1342 static const ssize_t
1343 X[4] = {0, 1, 1,-1},
1344 Y[4] = {1, 0, 1, 1};
1349 assert(image != (
const Image *) NULL);
1350 assert(image->signature == MagickCoreSignature);
1351 assert(exception != (ExceptionInfo *) NULL);
1352 assert(exception->signature == MagickCoreSignature);
1353 if (IsEventLogging() != MagickFalse)
1354 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
1355#if defined(MAGICKCORE_OPENCL_SUPPORT)
1356 despeckle_image=AccelerateDespeckleImage(image,exception);
1357 if (despeckle_image != (Image *) NULL)
1358 return(despeckle_image);
1360 despeckle_image=CloneImage(image,0,0,MagickTrue,exception);
1361 if (despeckle_image == (Image *) NULL)
1362 return((Image *) NULL);
1363 status=SetImageStorageClass(despeckle_image,DirectClass,exception);
1364 if (status == MagickFalse)
1366 despeckle_image=DestroyImage(despeckle_image);
1367 return((Image *) NULL);
1372 if ((image->columns > (MAGICK_SIZE_MAX-2)) ||
1373 (image->rows > (MAGICK_SIZE_MAX-2)))
1375 despeckle_image=DestroyImage(despeckle_image);
1376 ThrowImageException(ResourceLimitError,
"MemoryAllocationFailed");
1378 length=(image->columns+2)*(image->rows+2);
1379 pixel_info=AcquireVirtualMemory(length,
sizeof(*pixels));
1380 buffer_info=AcquireVirtualMemory(length,
sizeof(*buffer));
1381 if ((pixel_info == (MemoryInfo *) NULL) ||
1382 (buffer_info == (MemoryInfo *) NULL))
1384 if (buffer_info != (MemoryInfo *) NULL)
1385 buffer_info=RelinquishVirtualMemory(buffer_info);
1386 if (pixel_info != (MemoryInfo *) NULL)
1387 pixel_info=RelinquishVirtualMemory(pixel_info);
1388 despeckle_image=DestroyImage(despeckle_image);
1389 ThrowImageException(ResourceLimitError,
"MemoryAllocationFailed");
1391 pixels=(Quantum *) GetVirtualMemoryBlob(pixel_info);
1392 buffer=(Quantum *) GetVirtualMemoryBlob(buffer_info);
1397 image_view=AcquireVirtualCacheView(image,exception);
1398 despeckle_view=AcquireAuthenticCacheView(despeckle_image,exception);
1399 for (i=0; i < (ssize_t) GetPixelChannels(image); i++)
1416 if (status == MagickFalse)
1418 channel=GetPixelChannelChannel(image,i);
1419 traits=GetPixelChannelTraits(image,channel);
1420 despeckle_traits=GetPixelChannelTraits(despeckle_image,channel);
1421 if ((traits == UndefinedPixelTrait) ||
1422 (despeckle_traits == UndefinedPixelTrait))
1424 if ((despeckle_traits & CopyPixelTrait) != 0)
1426 (void) memset(pixels,0,length*
sizeof(*pixels));
1427 j=(ssize_t) image->columns+2;
1428 for (y=0; y < (ssize_t) image->rows; y++)
1433 p=GetCacheViewVirtualPixels(image_view,0,y,image->columns,1,exception);
1434 if (p == (
const Quantum *) NULL)
1440 for (x=0; x < (ssize_t) image->columns; x++)
1443 p+=(ptrdiff_t) GetPixelChannels(image);
1447 (void) memset(buffer,0,length*
sizeof(*buffer));
1448 for (k=0; k < 4; k++)
1450 Hull(image,X[k],Y[k],image->columns,image->rows,1,pixels,buffer);
1451 Hull(image,-X[k],-Y[k],image->columns,image->rows,1,pixels,buffer);
1452 Hull(image,-X[k],-Y[k],image->columns,image->rows,-1,pixels,buffer);
1453 Hull(image,X[k],Y[k],image->columns,image->rows,-1,pixels,buffer);
1455 j=(ssize_t) image->columns+2;
1456 for (y=0; y < (ssize_t) image->rows; y++)
1464 q=GetCacheViewAuthenticPixels(despeckle_view,0,y,despeckle_image->columns,
1466 if (q == (Quantum *) NULL)
1472 for (x=0; x < (ssize_t) image->columns; x++)
1474 SetPixelChannel(despeckle_image,channel,pixels[j++],q);
1475 q+=(ptrdiff_t) GetPixelChannels(despeckle_image);
1477 sync=SyncCacheViewAuthenticPixels(despeckle_view,exception);
1478 if (sync == MagickFalse)
1482 if (image->progress_monitor != (MagickProgressMonitor) NULL)
1487 proceed=SetImageProgress(image,DespeckleImageTag,(MagickOffsetType) i,
1488 GetPixelChannels(image));
1489 if (proceed == MagickFalse)
1493 despeckle_view=DestroyCacheView(despeckle_view);
1494 image_view=DestroyCacheView(image_view);
1495 buffer_info=RelinquishVirtualMemory(buffer_info);
1496 pixel_info=RelinquishVirtualMemory(pixel_info);
1497 despeckle_image->type=image->type;
1498 if (status == MagickFalse)
1499 despeckle_image=DestroyImage(despeckle_image);
1500 return(despeckle_image);
1532MagickExport Image *EdgeImage(
const Image *image,
const double radius,
1533 ExceptionInfo *exception)
1547 assert(image != (
const Image *) NULL);
1548 assert(image->signature == MagickCoreSignature);
1549 assert(exception != (ExceptionInfo *) NULL);
1550 assert(exception->signature == MagickCoreSignature);
1551 if (IsEventLogging() != MagickFalse)
1552 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
1553 width=GetOptimalKernelWidth1D(radius,0.5);
1554 kernel_info=AcquireKernelInfo((
const char *) NULL,exception);
1555 if (kernel_info == (KernelInfo *) NULL)
1556 ThrowImageException(ResourceLimitError,
"MemoryAllocationFailed");
1557 (void) memset(kernel_info,0,
sizeof(*kernel_info));
1558 kernel_info->width=width;
1559 kernel_info->height=width;
1560 kernel_info->x=(ssize_t) (kernel_info->width-1)/2;
1561 kernel_info->y=(ssize_t) (kernel_info->height-1)/2;
1562 kernel_info->signature=MagickCoreSignature;
1563 kernel_info->values=(MagickRealType *) MagickAssumeAligned(
1564 AcquireAlignedMemory(kernel_info->width,kernel_info->height*
1565 sizeof(*kernel_info->values)));
1566 if (kernel_info->values == (MagickRealType *) NULL)
1568 kernel_info=DestroyKernelInfo(kernel_info);
1569 ThrowImageException(ResourceLimitError,
"MemoryAllocationFailed");
1571 for (i=0; i < (ssize_t) (kernel_info->width*kernel_info->height); i++)
1572 kernel_info->values[i]=(-1.0);
1573 kernel_info->values[i/2]=(double) kernel_info->width*kernel_info->height-1.0;
1574 edge_image=ConvolveImage(image,kernel_info,exception);
1575 kernel_info=DestroyKernelInfo(kernel_info);
1612MagickExport Image *EmbossImage(
const Image *image,
const double radius,
1613 const double sigma,ExceptionInfo *exception)
1637 assert(image != (
const Image *) NULL);
1638 assert(image->signature == MagickCoreSignature);
1639 assert(exception != (ExceptionInfo *) NULL);
1640 assert(exception->signature == MagickCoreSignature);
1641 if (IsEventLogging() != MagickFalse)
1642 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
1643 width=GetOptimalKernelWidth1D(radius,sigma);
1644 kernel_info=AcquireKernelInfo((
const char *) NULL,exception);
1645 if (kernel_info == (KernelInfo *) NULL)
1646 ThrowImageException(ResourceLimitError,
"MemoryAllocationFailed");
1647 kernel_info->width=width;
1648 kernel_info->height=width;
1649 kernel_info->x=(ssize_t) (width-1)/2;
1650 kernel_info->y=(ssize_t) (width-1)/2;
1651 kernel_info->values=(MagickRealType *) MagickAssumeAligned(
1652 AcquireAlignedMemory(kernel_info->width,kernel_info->width*
1653 sizeof(*kernel_info->values)));
1654 if (kernel_info->values == (MagickRealType *) NULL)
1656 kernel_info=DestroyKernelInfo(kernel_info);
1657 ThrowImageException(ResourceLimitError,
"MemoryAllocationFailed");
1659 j=(ssize_t) (kernel_info->width-1)/2;
1662 for (v=(-j); v <= j; v++)
1664 for (u=(-j); u <= j; u++)
1666 kernel_info->values[i]=(MagickRealType) (((u < 0) || (v < 0) ? -8.0 :
1667 8.0)*exp(-((
double) u*u+v*v)/(2.0*MagickSigma*MagickSigma))/
1668 (2.0*MagickPI*MagickSigma*MagickSigma));
1670 kernel_info->values[i]=0.0;
1676 for (i=0; i < (ssize_t) (kernel_info->width*kernel_info->height); i++)
1677 normalize+=kernel_info->values[i];
1678 gamma=MagickSafeReciprocal(normalize);
1679 for (i=0; i < (ssize_t) (kernel_info->width*kernel_info->height); i++)
1680 kernel_info->values[i]*=gamma;
1681 emboss_image=ConvolveImage(image,kernel_info,exception);
1682 kernel_info=DestroyKernelInfo(kernel_info);
1683 if (emboss_image != (Image *) NULL)
1684 (void) EqualizeImage(emboss_image,exception);
1685 return(emboss_image);
1721MagickExport Image *GaussianBlurImage(
const Image *image,
const double radius,
1722 const double sigma,ExceptionInfo *exception)
1725 geometry[MagickPathExtent];
1733 assert(image != (
const Image *) NULL);
1734 assert(image->signature == MagickCoreSignature);
1735 assert(exception != (ExceptionInfo *) NULL);
1736 assert(exception->signature == MagickCoreSignature);
1737 if (IsEventLogging() != MagickFalse)
1738 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
1739 (void) FormatLocaleString(geometry,MagickPathExtent,
"gaussian:%.20gx%.20g",
1741 kernel_info=AcquireKernelInfo(geometry,exception);
1742 if (kernel_info == (KernelInfo *) NULL)
1743 ThrowImageException(ResourceLimitError,
"MemoryAllocationFailed");
1744 blur_image=ConvolveImage(image,kernel_info,exception);
1745 kernel_info=DestroyKernelInfo(kernel_info);
1779static inline MagickRealType GetMeanLuma(
const Image *magick_restrict image,
1780 const double *magick_restrict pixel)
1782 return(0.212656*pixel[image->channel_map[RedPixelChannel].offset]+
1783 0.715158*pixel[image->channel_map[GreenPixelChannel].offset]+
1784 0.072186*pixel[image->channel_map[BluePixelChannel].offset]);
1787MagickExport Image *KuwaharaImage(
const Image *image,
const double radius,
1788 const double sigma,ExceptionInfo *exception)
1790#define KuwaharaImageTag "Kuwahara/Image"
1815 assert(image != (Image *) NULL);
1816 assert(image->signature == MagickCoreSignature);
1817 assert(exception != (ExceptionInfo *) NULL);
1818 assert(exception->signature == MagickCoreSignature);
1819 if (IsEventLogging() != MagickFalse)
1820 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
1821 width=(size_t) radius+1;
1822 gaussian_image=BlurImage(image,radius,sigma,exception);
1823 if (gaussian_image == (Image *) NULL)
1824 return((Image *) NULL);
1825 kuwahara_image=CloneImage(image,0,0,MagickTrue,exception);
1826 if (kuwahara_image == (Image *) NULL)
1828 gaussian_image=DestroyImage(gaussian_image);
1829 return((Image *) NULL);
1831 if (SetImageStorageClass(kuwahara_image,DirectClass,exception) == MagickFalse)
1833 gaussian_image=DestroyImage(gaussian_image);
1834 kuwahara_image=DestroyImage(kuwahara_image);
1835 return((Image *) NULL);
1842 image_view=AcquireVirtualCacheView(gaussian_image,exception);
1843 kuwahara_view=AcquireAuthenticCacheView(kuwahara_image,exception);
1844#if defined(MAGICKCORE_OPENMP_SUPPORT)
1845 #pragma omp parallel for schedule(static) shared(progress,status) \
1846 magick_number_threads(image,kuwahara_image,gaussian_image->rows,1)
1848 for (y=0; y < (ssize_t) gaussian_image->rows; y++)
1856 if (status == MagickFalse)
1858 q=QueueCacheViewAuthenticPixels(kuwahara_view,0,y,kuwahara_image->columns,1,
1860 if (q == (Quantum *) NULL)
1865 for (x=0; x < (ssize_t) gaussian_image->columns; x++)
1880 min_variance=MagickMaximumValue;
1881 SetGeometry(gaussian_image,&target);
1882 quadrant.width=width;
1883 quadrant.height=width;
1884 for (i=0; i < 4; i++)
1890 mean[MaxPixelChannels],
1905 quadrant.x=x-(ssize_t) (width-1);
1906 quadrant.y=y-(ssize_t) (width-1);
1911 quadrant.y=y-(ssize_t) (width-1);
1916 quadrant.x=x-(ssize_t) (width-1);
1923 p=GetCacheViewVirtualPixels(image_view,quadrant.x,quadrant.y,
1924 quadrant.width,quadrant.height,exception);
1925 if (p == (
const Quantum *) NULL)
1927 for (j=0; j < (ssize_t) GetPixelChannels(gaussian_image); j++)
1930 for (n=0; n < (ssize_t) (width*width); n++)
1932 for (j=0; j < (ssize_t) GetPixelChannels(gaussian_image); j++)
1933 mean[j]+=(
double) k[j];
1934 k+=(ptrdiff_t) GetPixelChannels(gaussian_image);
1936 for (j=0; j < (ssize_t) GetPixelChannels(gaussian_image); j++)
1937 mean[j]/=(
double) (width*width);
1940 for (n=0; n < (ssize_t) (width*width); n++)
1945 luma=GetPixelLuma(gaussian_image,k);
1946 variance+=(luma-GetMeanLuma(gaussian_image,mean))*
1947 (luma-GetMeanLuma(gaussian_image,mean));
1948 k+=(ptrdiff_t) GetPixelChannels(gaussian_image);
1950 if (variance < min_variance)
1952 min_variance=variance;
1961 status=InterpolatePixelChannels(gaussian_image,image_view,kuwahara_image,
1962 UndefinedInterpolatePixel,(
double) target.x+target.width/2.0,(
double)
1963 target.y+target.height/2.0,q,exception);
1964 if (status == MagickFalse)
1966 q+=(ptrdiff_t) GetPixelChannels(kuwahara_image);
1968 if (SyncCacheViewAuthenticPixels(kuwahara_view,exception) == MagickFalse)
1970 if (image->progress_monitor != (MagickProgressMonitor) NULL)
1975#if defined(MAGICKCORE_OPENMP_SUPPORT)
1979 proceed=SetImageProgress(image,KuwaharaImageTag,progress,image->rows);
1980 if (proceed == MagickFalse)
1984 kuwahara_view=DestroyCacheView(kuwahara_view);
1985 image_view=DestroyCacheView(image_view);
1986 gaussian_image=DestroyImage(gaussian_image);
1987 if (status == MagickFalse)
1988 kuwahara_image=DestroyImage(kuwahara_image);
1989 return(kuwahara_image);
2025MagickExport Image *LocalContrastImage(
const Image *image,
const double radius,
2026 const double strength,ExceptionInfo *exception)
2028#define LocalContrastImageTag "LocalContrast/Image"
2058 assert(image != (
const Image *) NULL);
2059 assert(image->signature == MagickCoreSignature);
2060 assert(exception != (ExceptionInfo *) NULL);
2061 assert(exception->signature == MagickCoreSignature);
2062 if (IsEventLogging() != MagickFalse)
2063 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
2064#if defined(MAGICKCORE_OPENCL_SUPPORT)
2065 contrast_image=AccelerateLocalContrastImage(image,radius,strength,exception);
2066 if (contrast_image != (Image *) NULL)
2067 return(contrast_image);
2069 contrast_image=CloneImage(image,0,0,MagickTrue,exception);
2070 if (contrast_image == (Image *) NULL)
2071 return((Image *) NULL);
2072 if (SetImageStorageClass(contrast_image,DirectClass,exception) == MagickFalse)
2074 contrast_image=DestroyImage(contrast_image);
2075 return((Image *) NULL);
2077 image_view=AcquireVirtualCacheView(image,exception);
2078 contrast_view=AcquireAuthenticCacheView(contrast_image,exception);
2079 scanLineSize=(ssize_t) MagickMax(image->columns,image->rows);
2080 width=(ssize_t) (scanLineSize*0.002*fabs(radius));
2081 scanLineSize+=(2*width);
2082 scanline_info=AcquireVirtualMemory(GetOpenMPMaximumThreads()*
2083 (
size_t) scanLineSize,
sizeof(*scanline));
2084 if (scanline_info == (MemoryInfo *) NULL)
2086 contrast_view=DestroyCacheView(contrast_view);
2087 image_view=DestroyCacheView(image_view);
2088 contrast_image=DestroyImage(contrast_image);
2089 ThrowImageException(ResourceLimitError,
"MemoryAllocationFailed");
2091 scanline=(
float *) GetVirtualMemoryBlob(scanline_info);
2095 interImage_info=AcquireVirtualMemory(image->rows*(image->columns+(
size_t)
2096 (2*width)),
sizeof(*interImage));
2097 if (interImage_info == (MemoryInfo *) NULL)
2099 scanline_info=RelinquishVirtualMemory(scanline_info);
2100 contrast_view=DestroyCacheView(contrast_view);
2101 image_view=DestroyCacheView(image_view);
2102 contrast_image=DestroyImage(contrast_image);
2103 ThrowImageException(ResourceLimitError,
"MemoryAllocationFailed");
2105 interImage=(
float *) GetVirtualMemoryBlob(interImage_info);
2106 totalWeight=(float) ((width+1)*(width+1));
2115#if defined(MAGICKCORE_OPENMP_SUPPORT)
2116#pragma omp parallel for schedule(static) \
2117 magick_number_threads(image,image,image->columns,1)
2119 for (x=0; x < (ssize_t) image->columns; x++)
2122 id = GetOpenMPThreadId();
2138 if (status == MagickFalse)
2141 pixels+=
id*scanLineSize;
2143 p=GetCacheViewVirtualPixels(image_view,x,-(ssize_t) width,1,
2144 image->rows+(
size_t) (2*width),exception);
2145 if (p == (
const Quantum *) NULL)
2150 for (y=0; y < (ssize_t) image->rows+(2*width); y++)
2152 *pix++=(float)GetPixelLuma(image,p);
2153 p+=(ptrdiff_t) image->number_channels;
2155 out=interImage+x+width;
2156 for (y=0; y < (ssize_t) image->rows; y++)
2165 for (i=0; i < width; i++)
2167 sum+=weight*((double) *pix++);
2170 for (i=width+1; i < (2*width); i++)
2172 sum+=weight*((double) *pix++);
2176 *out=(float) (sum/totalWeight);
2178 if ((x <= width) && (x != 0))
2180 if ((x > (ssize_t) image->columns-width-2) &&
2181 (x != (ssize_t) image->columns-1))
2182 *(out+((image->columns-(size_t) x-1)*2))=*out;
2183 out+=image->columns+(size_t) (width*2);
2194#if defined(MAGICKCORE_OPENMP_SUPPORT)
2195#pragma omp parallel for schedule(static) \
2196 magick_number_threads(image,image,image->rows,1)
2198 for (y=0; y < (ssize_t) image->rows; y++)
2201 id = GetOpenMPThreadId();
2217 if (status == MagickFalse)
2220 pixels+=
id*scanLineSize;
2221 p=GetCacheViewVirtualPixels(image_view,0,y,image->columns,1,exception);
2222 q=GetCacheViewAuthenticPixels(contrast_view,0,y,image->columns,1,
2224 if ((p == (
const Quantum *) NULL) || (q == (Quantum *) NULL))
2229 memcpy(pixels,interImage+((
size_t) y*(image->columns+(
size_t) (2*width))),
2230 (image->columns+(
size_t) (2*width))*
sizeof(
float));
2231 for (x=0; x < (ssize_t) image->columns; x++)
2245 for (i=0; i < width; i++)
2247 sum+=weight*((double) *pix++);
2250 for (i=width+1; i < (2*width); i++)
2252 sum+=weight*((double) *pix++);
2258 srcVal=(float) GetPixelLuma(image,p);
2259 mult=(srcVal-(sum/totalWeight))*(strength/100.0);
2260 mult=(srcVal+mult)/srcVal;
2261 traits=GetPixelChannelTraits(image,RedPixelChannel);
2262 if ((traits & UpdatePixelTrait) != 0)
2263 SetPixelRed(contrast_image,ClampToQuantum((MagickRealType)
2264 GetPixelRed(image,p)*mult),q);
2265 traits=GetPixelChannelTraits(image,GreenPixelChannel);
2266 if ((traits & UpdatePixelTrait) != 0)
2267 SetPixelGreen(contrast_image,ClampToQuantum((MagickRealType)
2268 GetPixelGreen(image,p)*mult),q);
2269 traits=GetPixelChannelTraits(image,BluePixelChannel);
2270 if ((traits & UpdatePixelTrait) != 0)
2271 SetPixelBlue(contrast_image,ClampToQuantum((MagickRealType)
2272 GetPixelBlue(image,p)*mult),q);
2273 p+=(ptrdiff_t) image->number_channels;
2274 q+=(ptrdiff_t) contrast_image->number_channels;
2276 if (SyncCacheViewAuthenticPixels(contrast_view,exception) == MagickFalse)
2280 scanline_info=RelinquishVirtualMemory(scanline_info);
2281 interImage_info=RelinquishVirtualMemory(interImage_info);
2282 contrast_view=DestroyCacheView(contrast_view);
2283 image_view=DestroyCacheView(image_view);
2284 if (status == MagickFalse)
2285 contrast_image=DestroyImage(contrast_image);
2286 return(contrast_image);
2328static MagickRealType *GetMotionBlurKernel(
const size_t width,
2341 if (IsEventLogging() != MagickFalse)
2342 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"...");
2343 kernel=(MagickRealType *) MagickAssumeAligned(AcquireAlignedMemory((
size_t)
2344 width,
sizeof(*kernel)));
2345 if (kernel == (MagickRealType *) NULL)
2348 for (i=0; i < (ssize_t) width; i++)
2350 kernel[i]=(MagickRealType) (exp((-((
double) i*i)/(
double) (2.0*MagickSigma*
2351 MagickSigma)))/(MagickSQ2PI*MagickSigma));
2352 normalize+=kernel[i];
2354 for (i=0; i < (ssize_t) width; i++)
2355 kernel[i]/=normalize;
2359MagickExport Image *MotionBlurImage(
const Image *image,
const double radius,
2360 const double sigma,
const double angle,ExceptionInfo *exception)
2362#define BlurImageTag "Blur/Image"
2394 assert(image != (Image *) NULL);
2395 assert(image->signature == MagickCoreSignature);
2396 assert(exception != (ExceptionInfo *) NULL);
2397 assert(exception->signature == MagickCoreSignature);
2398 if (IsEventLogging() != MagickFalse)
2399 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
2400 width=GetOptimalKernelWidth1D(radius,sigma);
2401 kernel=GetMotionBlurKernel(width,sigma);
2402 if (kernel == (MagickRealType *) NULL)
2403 ThrowImageException(ResourceLimitError,
"MemoryAllocationFailed");
2404 offset=(OffsetInfo *) AcquireQuantumMemory(width,
sizeof(*offset));
2405 if (offset == (OffsetInfo *) NULL)
2407 kernel=(MagickRealType *) RelinquishAlignedMemory(kernel);
2408 ThrowImageException(ResourceLimitError,
"MemoryAllocationFailed");
2410 point.x=(double) width*sin(DegreesToRadians(angle));
2411 point.y=(double) width*cos(DegreesToRadians(angle));
2412 for (w=0; w < (ssize_t) width; w++)
2414 offset[w].x=CastDoubleToSsizeT(ceil((
double) (w*point.y)/
2415 hypot(point.x,point.y)-0.5));
2416 offset[w].y=CastDoubleToSsizeT(ceil((
double) (w*point.x)/
2417 hypot(point.x,point.y)-0.5));
2422#if defined(MAGICKCORE_OPENCL_SUPPORT)
2423 blur_image=AccelerateMotionBlurImage(image,kernel,width,offset,exception);
2424 if (blur_image != (Image *) NULL)
2426 kernel=(MagickRealType *) RelinquishAlignedMemory(kernel);
2427 offset=(OffsetInfo *) RelinquishMagickMemory(offset);
2431 blur_image=CloneImage(image,0,0,MagickTrue,exception);
2432 if (blur_image == (Image *) NULL)
2434 kernel=(MagickRealType *) RelinquishAlignedMemory(kernel);
2435 offset=(OffsetInfo *) RelinquishMagickMemory(offset);
2436 return((Image *) NULL);
2438 if (SetImageStorageClass(blur_image,DirectClass,exception) == MagickFalse)
2440 kernel=(MagickRealType *) RelinquishAlignedMemory(kernel);
2441 offset=(OffsetInfo *) RelinquishMagickMemory(offset);
2442 blur_image=DestroyImage(blur_image);
2443 return((Image *) NULL);
2447 image_view=AcquireVirtualCacheView(image,exception);
2448 motion_view=AcquireVirtualCacheView(image,exception);
2449 blur_view=AcquireAuthenticCacheView(blur_image,exception);
2450#if defined(MAGICKCORE_OPENMP_SUPPORT)
2451 #pragma omp parallel for schedule(static) shared(progress,status) \
2452 magick_number_threads(image,blur_image,image->rows,1)
2454 for (y=0; y < (ssize_t) image->rows; y++)
2465 if (status == MagickFalse)
2467 p=GetCacheViewVirtualPixels(image_view,0,y,image->columns,1,exception);
2468 q=QueueCacheViewAuthenticPixels(blur_view,0,y,blur_image->columns,1,
2470 if ((p == (
const Quantum *) NULL) || (q == (Quantum *) NULL))
2475 for (x=0; x < (ssize_t) image->columns; x++)
2480 for (i=0; i < (ssize_t) GetPixelChannels(image); i++)
2503 channel=GetPixelChannelChannel(image,i);
2504 traits=GetPixelChannelTraits(image,channel);
2505 blur_traits=GetPixelChannelTraits(blur_image,channel);
2506 if ((traits == UndefinedPixelTrait) ||
2507 (blur_traits == UndefinedPixelTrait))
2509 if ((blur_traits & CopyPixelTrait) != 0)
2511 SetPixelChannel(blur_image,channel,p[i],q);
2516 if ((blur_traits & BlendPixelTrait) == 0)
2518 for (j=0; j < (ssize_t) width; j++)
2520 r=GetCacheViewVirtualPixels(motion_view,x+offset[j].x,y+
2521 offset[j].y,1,1,exception);
2522 if (r == (
const Quantum *) NULL)
2527 pixel+=(*k)*(double) r[i];
2530 SetPixelChannel(blur_image,channel,ClampToQuantum(pixel),q);
2533 for (j=0; j < (ssize_t) width; j++)
2535 r=GetCacheViewVirtualPixels(motion_view,x+offset[j].x,y+offset[j].y,1,
2537 if (r == (
const Quantum *) NULL)
2542 alpha=QuantumScale*(double) GetPixelAlpha(image,r);
2543 pixel+=(*k)*alpha*(double) r[i];
2547 gamma=MagickSafeReciprocal(gamma);
2548 SetPixelChannel(blur_image,channel,ClampToQuantum(gamma*pixel),q);
2550 p+=(ptrdiff_t) GetPixelChannels(image);
2551 q+=(ptrdiff_t) GetPixelChannels(blur_image);
2553 if (SyncCacheViewAuthenticPixels(blur_view,exception) == MagickFalse)
2555 if (image->progress_monitor != (MagickProgressMonitor) NULL)
2560#if defined(MAGICKCORE_OPENMP_SUPPORT)
2564 proceed=SetImageProgress(image,BlurImageTag,progress,image->rows);
2565 if (proceed == MagickFalse)
2569 blur_view=DestroyCacheView(blur_view);
2570 motion_view=DestroyCacheView(motion_view);
2571 image_view=DestroyCacheView(image_view);
2572 kernel=(MagickRealType *) RelinquishAlignedMemory(kernel);
2573 offset=(OffsetInfo *) RelinquishMagickMemory(offset);
2574 if (status == MagickFalse)
2575 blur_image=DestroyImage(blur_image);
2609MagickExport Image *PreviewImage(
const Image *image,
const PreviewType preview,
2610 ExceptionInfo *exception)
2612#define NumberTiles 9
2613#define PreviewImageTag "Preview/Image"
2614#define DefaultPreviewGeometry "204x204+10+10"
2617 factor[MagickPathExtent],
2618 label[MagickPathExtent];
2660 assert(image != (Image *) NULL);
2661 assert(image->signature == MagickCoreSignature);
2662 if (IsEventLogging() != MagickFalse)
2663 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
2667 preview_info=AcquireImageInfo();
2668 SetGeometry(image,&geometry);
2669 (void) ParseMetaGeometry(DefaultPreviewGeometry,&geometry.x,&geometry.y,
2670 &geometry.width,&geometry.height);
2671 images=NewImageList();
2673 GetQuantizeInfo(&quantize_info);
2677 for (i=0; i < NumberTiles; i++)
2679 thumbnail=ThumbnailImage(image,geometry.width,geometry.height,exception);
2680 if (thumbnail == (Image *) NULL)
2682 (void) SetImageProgressMonitor(thumbnail,(MagickProgressMonitor) NULL,
2684 (void) SetImageProperty(thumbnail,
"label",DefaultTileLabel,exception);
2685 if (i == (NumberTiles/2))
2687 (void) QueryColorCompliance(
"#dfdfdf",AllCompliance,
2688 &thumbnail->matte_color,exception);
2689 AppendImageToList(&images,thumbnail);
2697 preview_image=RotateImage(thumbnail,degrees,exception);
2698 (void) FormatLocaleString(label,MagickPathExtent,
"rotate %g",degrees);
2704 preview_image=ShearImage(thumbnail,degrees,degrees,exception);
2705 (void) FormatLocaleString(label,MagickPathExtent,
"shear %gx%g",degrees,
2711 x=((i+1)*(ssize_t) thumbnail->columns)/NumberTiles;
2712 y=((i+1)*(ssize_t) thumbnail->rows)/NumberTiles;
2713 preview_image=RollImage(thumbnail,x,y,exception);
2714 (void) FormatLocaleString(label,MagickPathExtent,
"roll %+.20gx%+.20g",
2715 (
double) x,(
double) y);
2720 preview_image=CloneImage(thumbnail,0,0,MagickTrue,exception);
2721 if (preview_image == (Image *) NULL)
2723 (void) FormatLocaleString(factor,MagickPathExtent,
"100,100,%g",2.0*
2725 (void) ModulateImage(preview_image,factor,exception);
2726 (void) FormatLocaleString(label,MagickPathExtent,
"modulate %s",factor);
2729 case SaturationPreview:
2731 preview_image=CloneImage(thumbnail,0,0,MagickTrue,exception);
2732 if (preview_image == (Image *) NULL)
2734 (void) FormatLocaleString(factor,MagickPathExtent,
"100,%g",2.0*
2736 (void) ModulateImage(preview_image,factor,exception);
2737 (void) FormatLocaleString(label,MagickPathExtent,
"modulate %s",factor);
2740 case BrightnessPreview:
2742 preview_image=CloneImage(thumbnail,0,0,MagickTrue,exception);
2743 if (preview_image == (Image *) NULL)
2745 (void) FormatLocaleString(factor,MagickPathExtent,
"%g",2.0*percentage);
2746 (void) ModulateImage(preview_image,factor,exception);
2747 (void) FormatLocaleString(label,MagickPathExtent,
"modulate %s",factor);
2753 preview_image=CloneImage(thumbnail,0,0,MagickTrue,exception);
2754 if (preview_image == (Image *) NULL)
2757 (void) GammaImage(preview_image,gamma,exception);
2758 (void) FormatLocaleString(label,MagickPathExtent,
"gamma %g",gamma);
2763 preview_image=CloneImage(thumbnail,0,0,MagickTrue,exception);
2764 if (preview_image != (Image *) NULL)
2765 for (x=0; x < i; x++)
2766 (
void) ContrastImage(preview_image,MagickTrue,exception);
2767 (void) FormatLocaleString(label,MagickPathExtent,
"contrast (%.20g)",
2773 preview_image=CloneImage(thumbnail,0,0,MagickTrue,exception);
2774 if (preview_image == (Image *) NULL)
2776 for (x=0; x < i; x++)
2777 (
void) ContrastImage(preview_image,MagickFalse,exception);
2778 (void) FormatLocaleString(label,MagickPathExtent,
"+contrast (%.20g)",
2782 case GrayscalePreview:
2784 preview_image=CloneImage(thumbnail,0,0,MagickTrue,exception);
2785 if (preview_image == (Image *) NULL)
2788 quantize_info.number_colors=colors;
2789 quantize_info.colorspace=GRAYColorspace;
2790 (void) QuantizeImage(&quantize_info,preview_image,exception);
2791 (void) FormatLocaleString(label,MagickPathExtent,
2792 "-colorspace gray -colors %.20g",(
double) colors);
2795 case QuantizePreview:
2797 preview_image=CloneImage(thumbnail,0,0,MagickTrue,exception);
2798 if (preview_image == (Image *) NULL)
2801 quantize_info.number_colors=colors;
2802 (void) QuantizeImage(&quantize_info,preview_image,exception);
2803 (void) FormatLocaleString(label,MagickPathExtent,
"colors %.20g",
2807 case DespecklePreview:
2809 for (x=0; x < (i-1); x++)
2811 preview_image=DespeckleImage(thumbnail,exception);
2812 if (preview_image == (Image *) NULL)
2814 thumbnail=DestroyImage(thumbnail);
2815 thumbnail=preview_image;
2817 preview_image=DespeckleImage(thumbnail,exception);
2818 if (preview_image == (Image *) NULL)
2820 (void) FormatLocaleString(label,MagickPathExtent,
"despeckle (%.20g)",
2824 case ReduceNoisePreview:
2826 preview_image=StatisticImage(thumbnail,NonpeakStatistic,(
size_t)
2827 radius,(
size_t) radius,exception);
2828 (void) FormatLocaleString(label,MagickPathExtent,
"noise %g",radius);
2831 case AddNoisePreview:
2837 (void) CopyMagickString(factor,
"uniform",MagickPathExtent);
2842 (void) CopyMagickString(factor,
"gaussian",MagickPathExtent);
2847 (void) CopyMagickString(factor,
"multiplicative",MagickPathExtent);
2852 (void) CopyMagickString(factor,
"impulse",MagickPathExtent);
2857 (void) CopyMagickString(factor,
"laplacian",MagickPathExtent);
2862 (void) CopyMagickString(factor,
"Poisson",MagickPathExtent);
2867 (void) CopyMagickString(thumbnail->magick,
"NULL",MagickPathExtent);
2871 preview_image=StatisticImage(thumbnail,NonpeakStatistic,(
size_t) i,
2872 (
size_t) i,exception);
2873 (void) FormatLocaleString(label,MagickPathExtent,
"+noise %s",factor);
2876 case SharpenPreview:
2878 preview_image=SharpenImage(thumbnail,radius,sigma,exception);
2879 (void) FormatLocaleString(label,MagickPathExtent,
"sharpen %gx%g",
2885 preview_image=BlurImage(thumbnail,radius,sigma,exception);
2886 (void) FormatLocaleString(label,MagickPathExtent,
"blur %gx%g",radius,
2890 case ThresholdPreview:
2892 preview_image=CloneImage(thumbnail,0,0,MagickTrue,exception);
2893 if (preview_image == (Image *) NULL)
2895 (void) BilevelImage(thumbnail,(
double) (percentage*((
double)
2896 QuantumRange+1.0))/100.0,exception);
2897 (void) FormatLocaleString(label,MagickPathExtent,
"threshold %g",
2898 (
double) (percentage*((
double) QuantumRange+1.0))/100.0);
2901 case EdgeDetectPreview:
2903 preview_image=EdgeImage(thumbnail,radius,exception);
2904 (void) FormatLocaleString(label,MagickPathExtent,
"edge %g",radius);
2909 preview_image=SpreadImage(thumbnail,image->interpolate,radius,
2911 (void) FormatLocaleString(label,MagickPathExtent,
"spread %g",
2915 case SolarizePreview:
2917 preview_image=CloneImage(thumbnail,0,0,MagickTrue,exception);
2918 if (preview_image == (Image *) NULL)
2920 (void) SolarizeImage(preview_image,(
double) QuantumRange*percentage/
2922 (void) FormatLocaleString(label,MagickPathExtent,
"solarize %g",
2923 ((
double) QuantumRange*percentage)/100.0);
2929 preview_image=ShadeImage(thumbnail,MagickTrue,degrees,degrees,
2931 (void) FormatLocaleString(label,MagickPathExtent,
"shade %gx%g",degrees,
2940 preview_image=CloneImage(thumbnail,0,0,MagickTrue,exception);
2941 if (preview_image == (Image *) NULL)
2943 raise.width=(size_t) (2*i+2);
2944 raise.height=(size_t) (2*i+2);
2947 (void) RaiseImage(preview_image,&raise,MagickTrue,exception);
2948 (void) FormatLocaleString(label,MagickPathExtent,
2949 "raise %.20gx%.20g%+.20g%+.20g",(
double) raise.width,(
double)
2950 raise.height,(
double) raise.x,(
double) raise.y);
2953 case SegmentPreview:
2955 preview_image=CloneImage(thumbnail,0,0,MagickTrue,exception);
2956 if (preview_image == (Image *) NULL)
2959 (void) SegmentImage(preview_image,sRGBColorspace,MagickFalse,threshold,
2960 threshold,exception);
2961 (void) FormatLocaleString(label,MagickPathExtent,
"segment %gx%g",
2962 threshold,threshold);
2967 preview_image=SwirlImage(thumbnail,degrees,image->interpolate,
2969 (void) FormatLocaleString(label,MagickPathExtent,
"swirl %g",degrees);
2973 case ImplodePreview:
2976 preview_image=ImplodeImage(thumbnail,degrees,image->interpolate,
2978 (void) FormatLocaleString(label,MagickPathExtent,
"implode %g",degrees);
2984 preview_image=WaveImage(thumbnail,0.5*degrees,2.0*degrees,
2985 image->interpolate,exception);
2986 (void) FormatLocaleString(label,MagickPathExtent,
"wave %gx%g",0.5*
2987 degrees,2.0*degrees);
2990 case OilPaintPreview:
2992 preview_image=OilPaintImage(thumbnail,(
double) radius,(
double) sigma,
2994 (void) FormatLocaleString(label,MagickPathExtent,
"charcoal %gx%g",
2998 case CharcoalDrawingPreview:
3000 preview_image=CharcoalImage(thumbnail,(
double) radius,(
double) sigma,
3002 (void) FormatLocaleString(label,MagickPathExtent,
"charcoal %gx%g",
3009 filename[MagickPathExtent];
3017 preview_image=CloneImage(thumbnail,0,0,MagickTrue,exception);
3018 if (preview_image == (Image *) NULL)
3020 preview_info->quality=(size_t) percentage;
3021 (void) FormatLocaleString(factor,MagickPathExtent,
"%.20g",(
double)
3022 preview_info->quality);
3023 file=AcquireUniqueFileResource(filename);
3025 file=close_utf8(file)-1;
3026 (void) FormatLocaleString(preview_image->filename,MagickPathExtent,
3027 "jpeg:%s",filename);
3028 status=WriteImage(preview_info,preview_image,exception);
3029 if (status != MagickFalse)
3034 (void) CopyMagickString(preview_info->filename,
3035 preview_image->filename,MagickPathExtent);
3036 quality_image=ReadImage(preview_info,exception);
3037 if (quality_image != (Image *) NULL)
3039 preview_image=DestroyImage(preview_image);
3040 preview_image=quality_image;
3043 (void) RelinquishUniqueFileResource(preview_image->filename);
3044 if ((GetBlobSize(preview_image)/1024) >= 1024)
3045 (void) FormatLocaleString(label,MagickPathExtent,
"quality %s\n%gmb ",
3046 factor,(
double) ((MagickOffsetType) GetBlobSize(preview_image))/
3049 if (GetBlobSize(preview_image) >= 1024)
3050 (void) FormatLocaleString(label,MagickPathExtent,
3051 "quality %s\n%gkb ",factor,(
double) ((MagickOffsetType)
3052 GetBlobSize(preview_image))/1024.0);
3054 (
void) FormatLocaleString(label,MagickPathExtent,
3055 "quality %s\n%.20gb ",factor,(
double) ((MagickOffsetType)
3056 GetBlobSize(thumbnail)));
3060 thumbnail=DestroyImage(thumbnail);
3064 if (preview_image == (Image *) NULL)
3066 preview_image->alpha_trait=UndefinedPixelTrait;
3067 (void) DeleteImageProperty(preview_image,
"label");
3068 (void) SetImageProperty(preview_image,
"label",label,exception);
3069 AppendImageToList(&images,preview_image);
3070 proceed=SetImageProgress(image,PreviewImageTag,(MagickOffsetType) i,
3072 if (proceed == MagickFalse)
3075 if (images == (Image *) NULL)
3077 preview_info=DestroyImageInfo(preview_info);
3078 return((Image *) NULL);
3083 montage_info=CloneMontageInfo(preview_info,(MontageInfo *) NULL);
3084 (void) CopyMagickString(montage_info->filename,image->filename,
3086 montage_info->shadow=MagickTrue;
3087 (void) CloneString(&montage_info->tile,
"3x3");
3088 (void) CloneString(&montage_info->geometry,DefaultPreviewGeometry);
3089 (void) CloneString(&montage_info->frame,DefaultTileFrame);
3090 montage_image=MontageImages(images,montage_info,exception);
3091 montage_info=DestroyMontageInfo(montage_info);
3092 images=DestroyImageList(images);
3093 if (montage_image == (Image *) NULL)
3094 ThrowImageException(ResourceLimitError,
"MemoryAllocationFailed");
3095 if (montage_image->montage != (
char *) NULL)
3100 montage_image->montage=(
char *) RelinquishMagickMemory(
3101 montage_image->montage);
3102 if (image->directory != (
char *) NULL)
3103 montage_image->directory=(
char *) RelinquishMagickMemory(
3104 montage_image->directory);
3106 preview_info=DestroyImageInfo(preview_info);
3107 return(montage_image);
3141MagickExport Image *RotationalBlurImage(
const Image *image,
const double angle,
3142 ExceptionInfo *exception)
3178 assert(image != (Image *) NULL);
3179 assert(image->signature == MagickCoreSignature);
3180 assert(exception != (ExceptionInfo *) NULL);
3181 assert(exception->signature == MagickCoreSignature);
3182 if (IsEventLogging() != MagickFalse)
3183 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
3184#if defined(MAGICKCORE_OPENCL_SUPPORT)
3185 blur_image=AccelerateRotationalBlurImage(image,angle,exception);
3186 if (blur_image != (Image *) NULL)
3189 blur_image=CloneImage(image,0,0,MagickTrue,exception);
3190 if (blur_image == (Image *) NULL)
3191 return((Image *) NULL);
3192 if (SetImageStorageClass(blur_image,DirectClass,exception) == MagickFalse)
3194 blur_image=DestroyImage(blur_image);
3195 return((Image *) NULL);
3197 blur_center.x=(double) (image->columns-1)/2.0;
3198 blur_center.y=(double) (image->rows-1)/2.0;
3199 blur_radius=hypot(blur_center.x,blur_center.y);
3200 n=(size_t) fabs(4.0*DegreesToRadians(angle)*sqrt((
double) blur_radius)+2UL);
3201 theta=DegreesToRadians(angle)/(double) (n-1);
3202 cos_theta=(
double *) AcquireQuantumMemory((
size_t) n,
sizeof(*cos_theta));
3203 sin_theta=(
double *) AcquireQuantumMemory((
size_t) n,
sizeof(*sin_theta));
3204 if ((cos_theta == (
double *) NULL) || (sin_theta == (
double *) NULL))
3206 if (cos_theta != (
double *) NULL)
3207 cos_theta=(
double *) RelinquishMagickMemory(cos_theta);
3208 if (sin_theta != (
double *) NULL)
3209 sin_theta=(
double *) RelinquishMagickMemory(sin_theta);
3210 blur_image=DestroyImage(blur_image);
3211 ThrowImageException(ResourceLimitError,
"MemoryAllocationFailed");
3213 offset=theta*(double) (n-1)/2.0;
3214 for (w=0; w < (ssize_t) n; w++)
3216 cos_theta[w]=cos((
double) (theta*w-offset));
3217 sin_theta[w]=sin((
double) (theta*w-offset));
3224 image_view=AcquireVirtualCacheView(image,exception);
3225 radial_view=AcquireVirtualCacheView(image,exception);
3226 blur_view=AcquireAuthenticCacheView(blur_image,exception);
3227#if defined(MAGICKCORE_OPENMP_SUPPORT)
3228 #pragma omp parallel for schedule(static) shared(progress,status) \
3229 magick_number_threads(image,blur_image,image->rows,1)
3231 for (y=0; y < (ssize_t) image->rows; y++)
3242 if (status == MagickFalse)
3244 p=GetCacheViewVirtualPixels(image_view,0,y,image->columns,1,exception);
3245 q=QueueCacheViewAuthenticPixels(blur_view,0,y,blur_image->columns,1,
3247 if ((p == (
const Quantum *) NULL) || (q == (Quantum *) NULL))
3252 for (x=0; x < (ssize_t) image->columns; x++)
3266 center.x=(double) x-blur_center.x;
3267 center.y=(double) y-blur_center.y;
3268 radius=hypot((
double) center.x,center.y);
3273 step=(size_t) (blur_radius/radius);
3280 for (i=0; i < (ssize_t) GetPixelChannels(image); i++)
3299 channel=GetPixelChannelChannel(image,i);
3300 traits=GetPixelChannelTraits(image,channel);
3301 blur_traits=GetPixelChannelTraits(blur_image,channel);
3302 if ((traits == UndefinedPixelTrait) ||
3303 (blur_traits == UndefinedPixelTrait))
3305 if ((blur_traits & CopyPixelTrait) != 0)
3307 SetPixelChannel(blur_image,channel,p[i],q);
3312 if ((GetPixelChannelTraits(image,AlphaPixelChannel) == UndefinedPixelTrait) ||
3313 (channel == AlphaPixelChannel))
3315 for (j=0; j < (ssize_t) n; j+=(ssize_t) step)
3317 r=GetCacheViewVirtualPixels(radial_view, (ssize_t) (blur_center.x+
3318 center.x*cos_theta[j]-center.y*sin_theta[j]+0.5),(ssize_t)
3319 (blur_center.y+center.x*sin_theta[j]+center.y*cos_theta[j]+0.5),
3321 if (r == (
const Quantum *) NULL)
3326 pixel+=(double) r[i];
3329 gamma=MagickSafeReciprocal(gamma);
3330 SetPixelChannel(blur_image,channel,ClampToQuantum(gamma*pixel),q);
3333 for (j=0; j < (ssize_t) n; j+=(ssize_t) step)
3338 r=GetCacheViewVirtualPixels(radial_view, (ssize_t) (blur_center.x+
3339 center.x*cos_theta[j]-center.y*sin_theta[j]+0.5),(ssize_t)
3340 (blur_center.y+center.x*sin_theta[j]+center.y*cos_theta[j]+0.5),
3342 if (r == (
const Quantum *) NULL)
3347 alpha=QuantumScale*(double) GetPixelAlpha(image,r);
3348 pixel+=alpha*(double) r[i];
3351 gamma=MagickSafeReciprocal(gamma);
3352 SetPixelChannel(blur_image,channel,ClampToQuantum(gamma*pixel),q);
3354 p+=(ptrdiff_t) GetPixelChannels(image);
3355 q+=(ptrdiff_t) GetPixelChannels(blur_image);
3357 if (SyncCacheViewAuthenticPixels(blur_view,exception) == MagickFalse)
3359 if (image->progress_monitor != (MagickProgressMonitor) NULL)
3364#if defined(MAGICKCORE_OPENMP_SUPPORT)
3368 proceed=SetImageProgress(image,BlurImageTag,progress,image->rows);
3369 if (proceed == MagickFalse)
3373 blur_view=DestroyCacheView(blur_view);
3374 radial_view=DestroyCacheView(radial_view);
3375 image_view=DestroyCacheView(image_view);
3376 cos_theta=(
double *) RelinquishMagickMemory(cos_theta);
3377 sin_theta=(
double *) RelinquishMagickMemory(sin_theta);
3378 if (status == MagickFalse)
3379 blur_image=DestroyImage(blur_image);
3418MagickExport Image *SelectiveBlurImage(
const Image *image,
const double radius,
3419 const double sigma,
const double threshold,ExceptionInfo *exception)
3421#define SelectiveBlurImageTag "SelectiveBlur/Image"
3451 assert(image != (Image *) NULL);
3452 assert(image->signature == MagickCoreSignature);
3453 assert(exception != (ExceptionInfo *) NULL);
3454 assert(exception->signature == MagickCoreSignature);
3455 if (IsEventLogging() != MagickFalse)
3456 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
3457 width=GetOptimalKernelWidth1D(radius,sigma);
3458 kernel=(MagickRealType *) MagickAssumeAligned(AcquireAlignedMemory((
size_t)
3459 width,width*
sizeof(*kernel)));
3460 if (kernel == (MagickRealType *) NULL)
3461 ThrowImageException(ResourceLimitError,
"MemoryAllocationFailed");
3468 j=(ssize_t) (width-1)/2;
3470 for (v=(-j); v <= j; v++)
3475 for (u=(-j); u <= j; u++)
3476 kernel[i++]=(MagickRealType) (exp(-((
double) u*u+v*v)/(2.0*MagickSigma*
3477 MagickSigma))/(2.0*MagickPI*MagickSigma*MagickSigma));
3480 if (image->debug != MagickFalse)
3483 format[MagickPathExtent],
3486 const MagickRealType
3493 (void) LogMagickEvent(TransformEvent,GetMagickModule(),
3494 " SelectiveBlurImage with %.20gx%.20g kernel:",(
double) width,(
double)
3496 message=AcquireString(
"");
3498 for (v=0; v < (ssize_t) width; v++)
3501 (void) FormatLocaleString(format,MagickPathExtent,
"%.20g: ",(
double) v);
3502 (void) ConcatenateString(&message,format);
3503 for (u=0; u < (ssize_t) width; u++)
3505 (void) FormatLocaleString(format,MagickPathExtent,
"%+f ",(
double)
3507 (void) ConcatenateString(&message,format);
3509 (void) LogMagickEvent(TransformEvent,GetMagickModule(),
"%s",message);
3511 message=DestroyString(message);
3513 blur_image=CloneImage(image,0,0,MagickTrue,exception);
3514 if (blur_image == (Image *) NULL)
3515 return((Image *) NULL);
3516 if (SetImageStorageClass(blur_image,DirectClass,exception) == MagickFalse)
3518 blur_image=DestroyImage(blur_image);
3519 kernel=(MagickRealType *) RelinquishAlignedMemory(kernel);
3520 return((Image *) NULL);
3522 luminance_image=CloneImage(image,0,0,MagickTrue,exception);
3523 if (luminance_image == (Image *) NULL)
3525 blur_image=DestroyImage(blur_image);
3526 kernel=(MagickRealType *) RelinquishAlignedMemory(kernel);
3527 return((Image *) NULL);
3529 status=TransformImageColorspace(luminance_image,GRAYColorspace,exception);
3530 if (status == MagickFalse)
3532 luminance_image=DestroyImage(luminance_image);
3533 blur_image=DestroyImage(blur_image);
3534 kernel=(MagickRealType *) RelinquishAlignedMemory(kernel);
3535 return((Image *) NULL);
3542 center=(ssize_t) (GetPixelChannels(image)*(image->columns+width)*
3543 ((width-1)/2L)+GetPixelChannels(image)*((width-1)/2L));
3544 image_view=AcquireVirtualCacheView(image,exception);
3545 luminance_view=AcquireVirtualCacheView(luminance_image,exception);
3546 blur_view=AcquireAuthenticCacheView(blur_image,exception);
3547#if defined(MAGICKCORE_OPENMP_SUPPORT)
3548 #pragma omp parallel for schedule(static) shared(progress,status) \
3549 magick_number_threads(image,blur_image,image->rows,1)
3551 for (y=0; y < (ssize_t) image->rows; y++)
3569 if (status == MagickFalse)
3571 p=GetCacheViewVirtualPixels(image_view,-((ssize_t) (width-1)/2L),y-(ssize_t)
3572 ((width-1)/2L),image->columns+width,width,exception);
3573 l=GetCacheViewVirtualPixels(luminance_view,-((ssize_t) (width-1)/2L),y-
3574 (ssize_t) ((width-1)/2L),luminance_image->columns+width,width,exception);
3575 q=QueueCacheViewAuthenticPixels(blur_view,0,y,blur_image->columns,1,
3577 if ((p == (
const Quantum *) NULL) || (l == (
const Quantum *) NULL) ||
3578 (q == (Quantum *) NULL))
3583 for (x=0; x < (ssize_t) image->columns; x++)
3591 intensity=GetPixelIntensity(image,p+center);
3592 for (i=0; i < (ssize_t) GetPixelChannels(image); i++)
3606 const MagickRealType
3610 *magick_restrict luminance_pixels,
3611 *magick_restrict pixels;
3619 channel=GetPixelChannelChannel(image,i);
3620 traits=GetPixelChannelTraits(image,channel);
3621 blur_traits=GetPixelChannelTraits(blur_image,channel);
3622 if ((traits == UndefinedPixelTrait) ||
3623 (blur_traits == UndefinedPixelTrait))
3625 if ((blur_traits & CopyPixelTrait) != 0)
3627 SetPixelChannel(blur_image,channel,p[center+i],q);
3635 if ((blur_traits & BlendPixelTrait) == 0)
3637 for (v=0; v < (ssize_t) width; v++)
3639 for (u=0; u < (ssize_t) width; u++)
3641 contrast=GetPixelIntensity(luminance_image,luminance_pixels)-
3643 if (fabs(contrast) < threshold)
3645 pixel+=(*k)*(double) pixels[i];
3649 pixels+=(ptrdiff_t) GetPixelChannels(image);
3650 luminance_pixels+=(ptrdiff_t) GetPixelChannels(luminance_image);
3652 pixels+=(ptrdiff_t) GetPixelChannels(image)*image->columns;
3653 luminance_pixels+=(ptrdiff_t) GetPixelChannels(luminance_image)*
3654 luminance_image->columns;
3656 if (fabs((
double) gamma) < MagickEpsilon)
3658 SetPixelChannel(blur_image,channel,p[center+i],q);
3661 gamma=MagickSafeReciprocal(gamma);
3662 SetPixelChannel(blur_image,channel,ClampToQuantum(gamma*pixel),q);
3665 for (v=0; v < (ssize_t) width; v++)
3667 for (u=0; u < (ssize_t) width; u++)
3669 contrast=GetPixelIntensity(image,pixels)-intensity;
3670 if (fabs(contrast) < threshold)
3672 alpha=QuantumScale*(double) GetPixelAlpha(image,pixels);
3673 pixel+=(*k)*alpha*(double) pixels[i];
3677 pixels+=(ptrdiff_t) GetPixelChannels(image);
3678 luminance_pixels+=(ptrdiff_t) GetPixelChannels(luminance_image);
3680 pixels+=(ptrdiff_t) GetPixelChannels(image)*image->columns;
3681 luminance_pixels+=(ptrdiff_t) GetPixelChannels(luminance_image)*
3682 luminance_image->columns;
3684 if (fabs((
double) gamma) < MagickEpsilon)
3686 SetPixelChannel(blur_image,channel,p[center+i],q);
3689 gamma=MagickSafeReciprocal(gamma);
3690 SetPixelChannel(blur_image,channel,ClampToQuantum(gamma*pixel),q);
3692 p+=(ptrdiff_t) GetPixelChannels(image);
3693 l+=(ptrdiff_t) GetPixelChannels(luminance_image);
3694 q+=(ptrdiff_t) GetPixelChannels(blur_image);
3696 sync=SyncCacheViewAuthenticPixels(blur_view,exception);
3697 if (sync == MagickFalse)
3699 if (image->progress_monitor != (MagickProgressMonitor) NULL)
3704#if defined(MAGICKCORE_OPENMP_SUPPORT)
3708 proceed=SetImageProgress(image,SelectiveBlurImageTag,progress,
3710 if (proceed == MagickFalse)
3714 blur_image->type=image->type;
3715 blur_view=DestroyCacheView(blur_view);
3716 luminance_view=DestroyCacheView(luminance_view);
3717 image_view=DestroyCacheView(image_view);
3718 luminance_image=DestroyImage(luminance_image);
3719 kernel=(MagickRealType *) RelinquishAlignedMemory(kernel);
3720 if (status == MagickFalse)
3721 blur_image=DestroyImage(blur_image);
3757MagickExport Image *ShadeImage(
const Image *image,
const MagickBooleanType gray,
3758 const double azimuth,
const double elevation,ExceptionInfo *exception)
3760#define GetShadeIntensity(image,pixel) \
3761 ClampPixel(GetPixelIntensity((image),(pixel)))
3762#define ShadeImageTag "Shade/Image"
3787 assert(image != (
const Image *) NULL);
3788 assert(image->signature == MagickCoreSignature);
3789 assert(exception != (ExceptionInfo *) NULL);
3790 assert(exception->signature == MagickCoreSignature);
3791 if (IsEventLogging() != MagickFalse)
3792 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
3793 linear_image=CloneImage(image,0,0,MagickTrue,exception);
3794 shade_image=CloneImage(image,0,0,MagickTrue,exception);
3795 if ((linear_image == (Image *) NULL) || (shade_image == (Image *) NULL))
3797 if (linear_image != (Image *) NULL)
3798 linear_image=DestroyImage(linear_image);
3799 if (shade_image != (Image *) NULL)
3800 shade_image=DestroyImage(shade_image);
3801 return((Image *) NULL);
3803 if (SetImageStorageClass(shade_image,DirectClass,exception) == MagickFalse)
3805 linear_image=DestroyImage(linear_image);
3806 shade_image=DestroyImage(shade_image);
3807 return((Image *) NULL);
3812 light.x=(double) QuantumRange*cos(DegreesToRadians(azimuth))*
3813 cos(DegreesToRadians(elevation));
3814 light.y=(double) QuantumRange*sin(DegreesToRadians(azimuth))*
3815 cos(DegreesToRadians(elevation));
3816 light.z=(double) QuantumRange*sin(DegreesToRadians(elevation));
3822 image_view=AcquireVirtualCacheView(linear_image,exception);
3823 shade_view=AcquireAuthenticCacheView(shade_image,exception);
3824#if defined(MAGICKCORE_OPENMP_SUPPORT)
3825 #pragma omp parallel for schedule(static) shared(progress,status) \
3826 magick_number_threads(linear_image,shade_image,linear_image->rows,1)
3828 for (y=0; y < (ssize_t) linear_image->rows; y++)
3839 *magick_restrict center,
3841 *magick_restrict post,
3842 *magick_restrict pre;
3850 if (status == MagickFalse)
3852 p=GetCacheViewVirtualPixels(image_view,-1,y-1,linear_image->columns+2,3,
3854 q=QueueCacheViewAuthenticPixels(shade_view,0,y,shade_image->columns,1,
3856 if ((p == (
const Quantum *) NULL) || (q == (Quantum *) NULL))
3864 normal.z=2.0*(double) QuantumRange;
3865 for (x=0; x < (ssize_t) linear_image->columns; x++)
3873 pre=p+GetPixelChannels(linear_image);
3874 center=pre+(linear_image->columns+2)*GetPixelChannels(linear_image);
3875 post=center+(linear_image->columns+2)*GetPixelChannels(linear_image);
3877 GetShadeIntensity(linear_image,pre-GetPixelChannels(linear_image))+
3878 GetShadeIntensity(linear_image,center-GetPixelChannels(linear_image))+
3879 GetShadeIntensity(linear_image,post-GetPixelChannels(linear_image))-
3880 GetShadeIntensity(linear_image,pre+GetPixelChannels(linear_image))-
3881 GetShadeIntensity(linear_image,center+GetPixelChannels(linear_image))-
3882 GetShadeIntensity(linear_image,post+GetPixelChannels(linear_image)));
3884 GetShadeIntensity(linear_image,post-GetPixelChannels(linear_image))+
3885 GetShadeIntensity(linear_image,post)+
3886 GetShadeIntensity(linear_image,post+GetPixelChannels(linear_image))-
3887 GetShadeIntensity(linear_image,pre-GetPixelChannels(linear_image))-
3888 GetShadeIntensity(linear_image,pre)-
3889 GetShadeIntensity(linear_image,pre+GetPixelChannels(linear_image)));
3890 if ((fabs(normal.x) <= MagickEpsilon) &&
3891 (fabs(normal.y) <= MagickEpsilon))
3896 distance=normal.x*light.x+normal.y*light.y+normal.z*light.z;
3897 if (distance > MagickEpsilon)
3899 normal_distance=normal.x*normal.x+normal.y*normal.y+
3901 if (normal_distance > (MagickEpsilon*MagickEpsilon))
3902 shade=distance/sqrt((
double) normal_distance);
3905 for (i=0; i < (ssize_t) GetPixelChannels(linear_image); i++)
3914 channel=GetPixelChannelChannel(linear_image,i);
3915 traits=GetPixelChannelTraits(linear_image,channel);
3916 shade_traits=GetPixelChannelTraits(shade_image,channel);
3917 if ((traits == UndefinedPixelTrait) ||
3918 (shade_traits == UndefinedPixelTrait))
3920 if ((shade_traits & CopyPixelTrait) != 0)
3922 SetPixelChannel(shade_image,channel,center[i],q);
3925 if ((traits & UpdatePixelTrait) == 0)
3927 SetPixelChannel(shade_image,channel,center[i],q);
3930 if (gray != MagickFalse)
3932 SetPixelChannel(shade_image,channel,ClampToQuantum(shade),q);
3935 SetPixelChannel(shade_image,channel,ClampToQuantum(QuantumScale*
3936 shade*(
double) center[i]),q);
3938 p+=(ptrdiff_t) GetPixelChannels(linear_image);
3939 q+=(ptrdiff_t) GetPixelChannels(shade_image);
3941 if (SyncCacheViewAuthenticPixels(shade_view,exception) == MagickFalse)
3943 if (image->progress_monitor != (MagickProgressMonitor) NULL)
3948#if defined(MAGICKCORE_OPENMP_SUPPORT)
3952 proceed=SetImageProgress(image,ShadeImageTag,progress,image->rows);
3953 if (proceed == MagickFalse)
3957 shade_view=DestroyCacheView(shade_view);
3958 image_view=DestroyCacheView(image_view);
3959 linear_image=DestroyImage(linear_image);
3960 if (status == MagickFalse)
3961 shade_image=DestroyImage(shade_image);
3962 return(shade_image);
4003MagickExport Image *SharpenImage(
const Image *image,
const double radius,
4004 const double sigma,ExceptionInfo *exception)
4027 assert(image != (
const Image *) NULL);
4028 assert(image->signature == MagickCoreSignature);
4029 assert(exception != (ExceptionInfo *) NULL);
4030 assert(exception->signature == MagickCoreSignature);
4031 if (IsEventLogging() != MagickFalse)
4032 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
4033 width=GetOptimalKernelWidth2D(radius,sigma);
4034 kernel_info=AcquireKernelInfo((
const char *) NULL,exception);
4035 if (kernel_info == (KernelInfo *) NULL)
4036 ThrowImageException(ResourceLimitError,
"MemoryAllocationFailed");
4037 (void) memset(kernel_info,0,
sizeof(*kernel_info));
4038 kernel_info->width=width;
4039 kernel_info->height=width;
4040 kernel_info->x=(ssize_t) (width-1)/2;
4041 kernel_info->y=(ssize_t) (width-1)/2;
4042 kernel_info->signature=MagickCoreSignature;
4043 kernel_info->values=(MagickRealType *) MagickAssumeAligned(
4044 AcquireAlignedMemory(kernel_info->width,kernel_info->height*
4045 sizeof(*kernel_info->values)));
4046 if (kernel_info->values == (MagickRealType *) NULL)
4048 kernel_info=DestroyKernelInfo(kernel_info);
4049 ThrowImageException(ResourceLimitError,
"MemoryAllocationFailed");
4052 j=(ssize_t) (kernel_info->width-1)/2;
4054 for (v=(-j); v <= j; v++)
4056 for (u=(-j); u <= j; u++)
4058 kernel_info->values[i]=(MagickRealType) (-exp(-((
double) u*u+v*v)/(2.0*
4059 MagickSigma*MagickSigma))/(2.0*MagickPI*MagickSigma*MagickSigma));
4060 normalize+=kernel_info->values[i];
4064 kernel_info->values[i/2]=(double) ((-2.0)*normalize);
4066 for (i=0; i < (ssize_t) (kernel_info->width*kernel_info->height); i++)
4067 normalize+=kernel_info->values[i];
4068 gamma=MagickSafeReciprocal(normalize);
4069 for (i=0; i < (ssize_t) (kernel_info->width*kernel_info->height); i++)
4070 kernel_info->values[i]*=gamma;
4071 sharp_image=ConvolveImage(image,kernel_info,exception);
4072 kernel_info=DestroyKernelInfo(kernel_info);
4073 return(sharp_image);
4107MagickExport Image *SpreadImage(
const Image *image,
4108 const PixelInterpolateMethod method,
const double radius,
4109 ExceptionInfo *exception)
4111#define SpreadImageTag "Spread/Image"
4127 **magick_restrict random_info;
4135#if defined(MAGICKCORE_OPENMP_SUPPORT)
4143 assert(image != (Image *) NULL);
4144 assert(image->signature == MagickCoreSignature);
4145 assert(exception != (ExceptionInfo *) NULL);
4146 assert(exception->signature == MagickCoreSignature);
4147 if (IsEventLogging() != MagickFalse)
4148 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
4149 spread_image=CloneImage(image,0,0,MagickTrue,exception);
4150 if (spread_image == (Image *) NULL)
4151 return((Image *) NULL);
4152 if (SetImageStorageClass(spread_image,DirectClass,exception) == MagickFalse)
4154 spread_image=DestroyImage(spread_image);
4155 return((Image *) NULL);
4162 width=GetOptimalKernelWidth1D(radius,0.5);
4163 random_info=AcquireRandomInfoTLS();
4164 image_view=AcquireVirtualCacheView(image,exception);
4165 spread_view=AcquireAuthenticCacheView(spread_image,exception);
4166#if defined(MAGICKCORE_OPENMP_SUPPORT)
4167 key=GetRandomSecretKey(random_info[0]);
4168 #pragma omp parallel for schedule(static) shared(progress,status) \
4169 magick_number_threads(image,spread_image,image->rows,key == ~0UL)
4171 for (y=0; y < (ssize_t) image->rows; y++)
4174 id = GetOpenMPThreadId();
4182 if (status == MagickFalse)
4184 q=QueueCacheViewAuthenticPixels(spread_view,0,y,spread_image->columns,1,
4186 if (q == (Quantum *) NULL)
4191 for (x=0; x < (ssize_t) image->columns; x++)
4196 point.x=GetPseudoRandomValue(random_info[
id]);
4197 point.y=GetPseudoRandomValue(random_info[
id]);
4198 status=InterpolatePixelChannels(image,image_view,spread_image,method,
4199 (
double) x+width*(point.x-0.5),(
double) y+width*(point.y-0.5),q,
4201 if (status == MagickFalse)
4203 q+=(ptrdiff_t) GetPixelChannels(spread_image);
4205 if (SyncCacheViewAuthenticPixels(spread_view,exception) == MagickFalse)
4207 if (image->progress_monitor != (MagickProgressMonitor) NULL)
4212#if defined(MAGICKCORE_OPENMP_SUPPORT)
4216 proceed=SetImageProgress(image,SpreadImageTag,progress,image->rows);
4217 if (proceed == MagickFalse)
4221 spread_view=DestroyCacheView(spread_view);
4222 image_view=DestroyCacheView(image_view);
4223 random_info=DestroyRandomInfoTLS(random_info);
4224 if (status == MagickFalse)
4225 spread_image=DestroyImage(spread_image);
4226 return(spread_image);
4268MagickExport Image *UnsharpMaskImage(
const Image *image,
const double radius,
4269 const double sigma,
const double gain,
const double threshold,
4270 ExceptionInfo *exception)
4272#define SharpenImageTag "Sharpen/Image"
4293 assert(image != (
const Image *) NULL);
4294 assert(image->signature == MagickCoreSignature);
4295 assert(exception != (ExceptionInfo *) NULL);
4296 assert(exception->signature == MagickCoreSignature);
4297 if (IsEventLogging() != MagickFalse)
4298 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
4307 unsharp_image=BlurImage(image,radius,sigma,exception);
4308 if (unsharp_image == (Image *) NULL)
4309 return((Image *) NULL);
4310 quantum_threshold=(double) QuantumRange*threshold;
4316 image_view=AcquireVirtualCacheView(image,exception);
4317 unsharp_view=AcquireAuthenticCacheView(unsharp_image,exception);
4318#if defined(MAGICKCORE_OPENMP_SUPPORT)
4319 #pragma omp parallel for schedule(static) shared(progress,status) \
4320 magick_number_threads(image,unsharp_image,image->rows,1)
4322 for (y=0; y < (ssize_t) image->rows; y++)
4333 if (status == MagickFalse)
4335 p=GetCacheViewVirtualPixels(image_view,0,y,image->columns,1,exception);
4336 q=GetCacheViewAuthenticPixels(unsharp_view,0,y,unsharp_image->columns,1,
4338 if ((p == (
const Quantum *) NULL) || (q == (Quantum *) NULL))
4343 for (x=0; x < (ssize_t) image->columns; x++)
4348 for (i=0; i < (ssize_t) GetPixelChannels(image); i++)
4360 channel=GetPixelChannelChannel(image,i);
4361 traits=GetPixelChannelTraits(image,channel);
4362 unsharp_traits=GetPixelChannelTraits(unsharp_image,channel);
4363 if ((traits == UndefinedPixelTrait) ||
4364 (unsharp_traits == UndefinedPixelTrait))
4366 if ((unsharp_traits & CopyPixelTrait) != 0)
4368 SetPixelChannel(unsharp_image,channel,p[i],q);
4371 pixel=(double) p[i]-(
double) GetPixelChannel(unsharp_image,channel,q);
4372 if (fabs(2.0*pixel) < quantum_threshold)
4373 pixel=(double) p[i];
4375 pixel=(double) p[i]+gain*pixel;
4376 SetPixelChannel(unsharp_image,channel,ClampToQuantum(pixel),q);
4378 p+=(ptrdiff_t) GetPixelChannels(image);
4379 q+=(ptrdiff_t) GetPixelChannels(unsharp_image);
4381 if (SyncCacheViewAuthenticPixels(unsharp_view,exception) == MagickFalse)
4383 if (image->progress_monitor != (MagickProgressMonitor) NULL)
4388#if defined(MAGICKCORE_OPENMP_SUPPORT)
4392 proceed=SetImageProgress(image,SharpenImageTag,progress,image->rows);
4393 if (proceed == MagickFalse)
4397 unsharp_image->type=image->type;
4398 unsharp_view=DestroyCacheView(unsharp_view);
4399 image_view=DestroyCacheView(image_view);
4400 if (status == MagickFalse)
4401 unsharp_image=DestroyImage(unsharp_image);
4402 return(unsharp_image);