• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2   Copyright 1999-2020 ImageMagick Studio LLC, a non-profit organization
3   dedicated to making software imaging solutions freely available.
4 
5   You may not use this file except in compliance with the License.  You may
6   obtain a copy of the License at
7 
8     https://imagemagick.org/script/license.php
9 
10   Unless required by applicable law or agreed to in writing, software
11   distributed under the License is distributed on an "AS IS" BASIS,
12   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13   See the License for the specific language governing permissions and
14   limitations under the License.
15 
16   MagickCore private kernels for accelerated functions.
17 */
18 
19 #ifndef MAGICKCORE_ACCELERATE_KERNELS_PRIVATE_H
20 #define MAGICKCORE_ACCELERATE_KERNELS_PRIVATE_H
21 
22 #if defined(__cplusplus) || defined(c_plusplus)
23 extern "C" {
24 #endif
25 
26 #if defined(MAGICKCORE_OPENCL_SUPPORT)
27 
28 /*
29   Define declarations.
30 */
31 #define OPENCL_DEFINE(VAR,...)	"\n #""define " #VAR " " #__VA_ARGS__ " \n"
32 #define OPENCL_ELIF(...)	"\n #""elif " #__VA_ARGS__ " \n"
33 #define OPENCL_ELSE()		"\n #""else " " \n"
34 #define OPENCL_ENDIF()		"\n #""endif " " \n"
35 #define OPENCL_IF(...)		"\n #""if " #__VA_ARGS__ " \n"
36 #define STRINGIFY(...) #__VA_ARGS__ "\n"
37 
38 const char *accelerateKernels =
39 
40 /*
41   Define declarations.
42 */
43   OPENCL_DEFINE(SigmaUniform, (attenuate*0.015625f))
44   OPENCL_DEFINE(SigmaGaussian, (attenuate*0.015625f))
45   OPENCL_DEFINE(SigmaImpulse, (attenuate*0.1f))
46   OPENCL_DEFINE(SigmaLaplacian, (attenuate*0.0390625f))
47   OPENCL_DEFINE(SigmaMultiplicativeGaussian, (attenuate*0.5f))
48   OPENCL_DEFINE(SigmaPoisson, (attenuate*12.5f))
49   OPENCL_DEFINE(SigmaRandom, (attenuate))
50   OPENCL_DEFINE(TauGaussian, (attenuate*0.078125f))
51   OPENCL_DEFINE(MagickMax(x,y), (((x) > (y)) ? (x) : (y)))
52   OPENCL_DEFINE(MagickMin(x,y), (((x) < (y)) ? (x) : (y)))
53 
54 /*
55   Typedef declarations.
56 */
57   STRINGIFY(
58     typedef enum
59     {
60       UndefinedColorspace,
61       CMYColorspace,           /* negated linear RGB colorspace */
62       CMYKColorspace,          /* CMY with Black separation */
63       GRAYColorspace,          /* Single Channel greyscale (non-linear) image */
64       HCLColorspace,
65       HCLpColorspace,
66       HSBColorspace,
67       HSIColorspace,
68       HSLColorspace,
69       HSVColorspace,           /* alias for HSB */
70       HWBColorspace,
71       LabColorspace,
72       LCHColorspace,           /* alias for LCHuv */
73       LCHabColorspace,         /* Cylindrical (Polar) Lab */
74       LCHuvColorspace,         /* Cylindrical (Polar) Luv */
75       LogColorspace,
76       LMSColorspace,
77       LuvColorspace,
78       OHTAColorspace,
79       Rec601YCbCrColorspace,
80       Rec709YCbCrColorspace,
81       RGBColorspace,           /* Linear RGB colorspace */
82       scRGBColorspace,         /* ??? */
83       sRGBColorspace,          /* Default: non-linear sRGB colorspace */
84       TransparentColorspace,
85       xyYColorspace,
86       XYZColorspace,           /* IEEE Color Reference colorspace */
87       YCbCrColorspace,
88       YCCColorspace,
89       YDbDrColorspace,
90       YIQColorspace,
91       YPbPrColorspace,
92       YUVColorspace,
93       LinearGRAYColorspace     /* Single Channel greyscale (linear) image */
94     } ColorspaceType;
95   )
96 
97   STRINGIFY(
98     typedef enum
99     {
100       UndefinedCompositeOp,
101       AlphaCompositeOp,
102       AtopCompositeOp,
103       BlendCompositeOp,
104       BlurCompositeOp,
105       BumpmapCompositeOp,
106       ChangeMaskCompositeOp,
107       ClearCompositeOp,
108       ColorBurnCompositeOp,
109       ColorDodgeCompositeOp,
110       ColorizeCompositeOp,
111       CopyBlackCompositeOp,
112       CopyBlueCompositeOp,
113       CopyCompositeOp,
114       CopyCyanCompositeOp,
115       CopyGreenCompositeOp,
116       CopyMagentaCompositeOp,
117       CopyAlphaCompositeOp,
118       CopyRedCompositeOp,
119       CopyYellowCompositeOp,
120       DarkenCompositeOp,
121       DarkenIntensityCompositeOp,
122       DifferenceCompositeOp,
123       DisplaceCompositeOp,
124       DissolveCompositeOp,
125       DistortCompositeOp,
126       DivideDstCompositeOp,
127       DivideSrcCompositeOp,
128       DstAtopCompositeOp,
129       DstCompositeOp,
130       DstInCompositeOp,
131       DstOutCompositeOp,
132       DstOverCompositeOp,
133       ExclusionCompositeOp,
134       HardLightCompositeOp,
135       HardMixCompositeOp,
136       HueCompositeOp,
137       InCompositeOp,
138       IntensityCompositeOp,
139       LightenCompositeOp,
140       LightenIntensityCompositeOp,
141       LinearBurnCompositeOp,
142       LinearDodgeCompositeOp,
143       LinearLightCompositeOp,
144       LuminizeCompositeOp,
145       MathematicsCompositeOp,
146       MinusDstCompositeOp,
147       MinusSrcCompositeOp,
148       ModulateCompositeOp,
149       ModulusAddCompositeOp,
150       ModulusSubtractCompositeOp,
151       MultiplyCompositeOp,
152       NoCompositeOp,
153       OutCompositeOp,
154       OverCompositeOp,
155       OverlayCompositeOp,
156       PegtopLightCompositeOp,
157       PinLightCompositeOp,
158       PlusCompositeOp,
159       ReplaceCompositeOp,
160       SaturateCompositeOp,
161       ScreenCompositeOp,
162       SoftLightCompositeOp,
163       SrcAtopCompositeOp,
164       SrcCompositeOp,
165       SrcInCompositeOp,
166       SrcOutCompositeOp,
167       SrcOverCompositeOp,
168       ThresholdCompositeOp,
169       VividLightCompositeOp,
170       XorCompositeOp,
171       StereoCompositeOp
172     } CompositeOperator;
173   )
174 
175   STRINGIFY(
176     typedef enum
177     {
178       UndefinedFunction,
179       ArcsinFunction,
180       ArctanFunction,
181       PolynomialFunction,
182       SinusoidFunction
183     } MagickFunction;
184   )
185 
186   STRINGIFY(
187     typedef enum
188     {
189       UndefinedNoise,
190       UniformNoise,
191       GaussianNoise,
192       MultiplicativeGaussianNoise,
193       ImpulseNoise,
194       LaplacianNoise,
195       PoissonNoise,
196       RandomNoise
197     } NoiseType;
198   )
199 
200   STRINGIFY(
201     typedef enum
202     {
203       UndefinedPixelIntensityMethod = 0,
204       AveragePixelIntensityMethod,
205       BrightnessPixelIntensityMethod,
206       LightnessPixelIntensityMethod,
207       MSPixelIntensityMethod,
208       Rec601LumaPixelIntensityMethod,
209       Rec601LuminancePixelIntensityMethod,
210       Rec709LumaPixelIntensityMethod,
211       Rec709LuminancePixelIntensityMethod,
212       RMSPixelIntensityMethod
213     } PixelIntensityMethod;
214   )
215 
216   STRINGIFY(
217     typedef enum
218     {
219       BoxWeightingFunction = 0,
220       TriangleWeightingFunction,
221       CubicBCWeightingFunction,
222       HannWeightingFunction,
223       HammingWeightingFunction,
224       BlackmanWeightingFunction,
225       GaussianWeightingFunction,
226       QuadraticWeightingFunction,
227       JincWeightingFunction,
228       SincWeightingFunction,
229       SincFastWeightingFunction,
230       KaiserWeightingFunction,
231       WelchWeightingFunction,
232       BohmanWeightingFunction,
233       LagrangeWeightingFunction,
234       CosineWeightingFunction
235     } ResizeWeightingFunctionType;
236   )
237 
238   STRINGIFY(
239     typedef enum
240     {
241       UndefinedChannel = 0x0000,
242       RedChannel = 0x0001,
243       GrayChannel = 0x0001,
244       CyanChannel = 0x0001,
245       GreenChannel = 0x0002,
246       MagentaChannel = 0x0002,
247       BlueChannel = 0x0004,
248       YellowChannel = 0x0004,
249       BlackChannel = 0x0008,
250       AlphaChannel = 0x0010,
251       OpacityChannel = 0x0010,
252       IndexChannel = 0x0020,             /* Color Index Table? */
253       ReadMaskChannel = 0x0040,          /* Pixel is Not Readable? */
254       WriteMaskChannel = 0x0080,         /* Pixel is Write Protected? */
255       MetaChannel = 0x0100,              /* ???? */
256       CompositeChannels = 0x001F,
257       AllChannels = 0x7ffffff,
258       /*
259         Special purpose channel types.
260         FUTURE: are these needed any more - they are more like hacks
261         SyncChannels for example is NOT a real channel but a 'flag'
262         It really says -- "User has not defined channels"
263         Though it does have extra meaning in the "-auto-level" operator
264       */
265       TrueAlphaChannel = 0x0100, /* extract actual alpha channel from opacity */
266       RGBChannels = 0x0200,      /* set alpha from grayscale mask in RGB */
267       GrayChannels = 0x0400,
268       SyncChannels = 0x20000,    /* channels modified as a single unit */
269       DefaultChannels = AllChannels
270     } ChannelType;  /* must correspond to PixelChannel */
271   )
272 
273 /*
274   Helper functions.
275 */
276 
277 OPENCL_IF((MAGICKCORE_QUANTUM_DEPTH == 8))
278 
279   STRINGIFY(
280     static inline CLQuantum ScaleCharToQuantum(const unsigned char value)
281     {
282       return((CLQuantum) value);
283     }
284   )
285 
286 OPENCL_ELIF((MAGICKCORE_QUANTUM_DEPTH == 16))
287 
288   STRINGIFY(
289     static inline CLQuantum ScaleCharToQuantum(const unsigned char value)
290     {
291       return((CLQuantum) (257.0f*value));
292     }
293   )
294 
295 OPENCL_ELIF((MAGICKCORE_QUANTUM_DEPTH == 32))
296 
297   STRINGIFY(
298     static inline CLQuantum ScaleCharToQuantum(const unsigned char value)
299     {
300       return((CLQuantum) (16843009.0*value));
301     }
302   )
303 
304 OPENCL_ENDIF()
305 
306 OPENCL_IF((MAGICKCORE_HDRI_SUPPORT == 1))
307 
308   STRINGIFY(
309     static inline CLQuantum ClampToQuantum(const float value)
310       {
311         return (CLQuantum) clamp(value, 0.0f, QuantumRange);
312       }
313   )
314 
315 OPENCL_ELSE()
316 
317   STRINGIFY(
318     static inline CLQuantum ClampToQuantum(const float value)
319       {
320         return (CLQuantum) (clamp(value, 0.0f, QuantumRange) + 0.5f);
321       }
322   )
323 
324 OPENCL_ENDIF()
325 
326   STRINGIFY(
327     static inline int ClampToCanvas(const int offset,const int range)
328       {
329         return clamp(offset, (int)0, range-1);
330       }
331   )
332 
333   STRINGIFY(
334     static inline uint ScaleQuantumToMap(CLQuantum value)
335       {
336         if (value >= (CLQuantum) MaxMap)
337           return ((uint)MaxMap);
338         else
339           return ((uint)value);
340       }
341   )
342 
343   STRINGIFY(
344     static inline float PerceptibleReciprocal(const float x)
345     {
346       float sign = x < (float) 0.0 ? (float) -1.0 : (float) 1.0;
347       return((sign*x) >= MagickEpsilon ? (float) 1.0/x : sign*((float) 1.0/MagickEpsilon));
348     }
349   )
350 
351   STRINGIFY(
352   static inline float RoundToUnity(const float value)
353    {
354      return clamp(value,0.0f,1.0f);
355    }
356   )
357 
358   STRINGIFY(
359 
360   static inline unsigned int getPixelIndex(const unsigned int number_channels,
361     const unsigned int columns, const unsigned int x, const unsigned int y)
362   {
363     return (x * number_channels) + (y * columns * number_channels);
364   }
365 
366   static inline float getPixelRed(const __global CLQuantum *p)   { return (float)*p; }
367   static inline float getPixelGreen(const __global CLQuantum *p) { return (float)*(p+1); }
368   static inline float getPixelBlue(const __global CLQuantum *p)  { return (float)*(p+2); }
369   static inline float getPixelAlpha(const __global CLQuantum *p,const unsigned int number_channels) { return (float)*(p+number_channels-1); }
370 
371   static inline void setPixelRed(__global CLQuantum *p,const CLQuantum value)   { *p=value; }
372   static inline void setPixelGreen(__global CLQuantum *p,const CLQuantum value) { *(p+1)=value; }
373   static inline void setPixelBlue(__global CLQuantum *p,const CLQuantum value)  { *(p+2)=value; }
374   static inline void setPixelAlpha(__global CLQuantum *p,const unsigned int number_channels,const CLQuantum value) { *(p+number_channels-1)=value; }
375 
376   static inline CLQuantum getBlue(CLPixelType p)               { return p.x; }
377   static inline void setBlue(CLPixelType* p, CLQuantum value)  { (*p).x = value; }
378   static inline float getBlueF4(float4 p)                      { return p.x; }
379   static inline void setBlueF4(float4* p, float value)         { (*p).x = value; }
380 
381   static inline CLQuantum getGreen(CLPixelType p)              { return p.y; }
382   static inline void setGreen(CLPixelType* p, CLQuantum value) { (*p).y = value; }
383   static inline float getGreenF4(float4 p)                     { return p.y; }
384   static inline void setGreenF4(float4* p, float value)        { (*p).y = value; }
385 
386   static inline CLQuantum getRed(CLPixelType p)                { return p.z; }
387   static inline void setRed(CLPixelType* p, CLQuantum value)   { (*p).z = value; }
388   static inline float getRedF4(float4 p)                       { return p.z; }
389   static inline void setRedF4(float4* p, float value)          { (*p).z = value; }
390 
391   static inline CLQuantum getAlpha(CLPixelType p)              { return p.w; }
392   static inline void setAlpha(CLPixelType* p, CLQuantum value) { (*p).w = value; }
393   static inline float getAlphaF4(float4 p)                     { return p.w; }
394   static inline void setAlphaF4(float4* p, float value)        { (*p).w = value; }
395 
396   static inline void ReadChannels(const __global CLQuantum *p, const unsigned int number_channels,
397     const ChannelType channel, float *red, float *green, float *blue, float *alpha)
398   {
399     if ((channel & RedChannel) != 0)
400       *red=getPixelRed(p);
401 
402     if (number_channels > 2)
403       {
404         if ((channel & GreenChannel) != 0)
405           *green=getPixelGreen(p);
406 
407         if ((channel & BlueChannel) != 0)
408           *blue=getPixelBlue(p);
409       }
410 
411     if (((number_channels == 4) || (number_channels == 2)) &&
412         ((channel & AlphaChannel) != 0))
413       *alpha=getPixelAlpha(p,number_channels);
414   }
415 
416   static inline float4 ReadAllChannels(const __global CLQuantum *image, const unsigned int number_channels,
417     const unsigned int columns, const unsigned int x, const unsigned int y)
418   {
419     const __global CLQuantum *p = image + getPixelIndex(number_channels, columns, x, y);
420 
421     float4 pixel;
422 
423     pixel.x=getPixelRed(p);
424 
425     if (number_channels > 2)
426       {
427         pixel.y=getPixelGreen(p);
428         pixel.z=getPixelBlue(p);
429       }
430 
431     if ((number_channels == 4) || (number_channels == 2))
432       pixel.w=getPixelAlpha(p,number_channels);
433     return(pixel);
434   }
435 
436   static inline float4 ReadFloat4(const __global CLQuantum *image, const unsigned int number_channels,
437     const unsigned int columns, const unsigned int x, const unsigned int y, const ChannelType channel)
438   {
439     const __global CLQuantum *p = image + getPixelIndex(number_channels, columns, x, y);
440 
441     float red = 0.0f;
442     float green = 0.0f;
443     float blue = 0.0f;
444     float alpha = 0.0f;
445 
446     ReadChannels(p, number_channels, channel, &red, &green, &blue, &alpha);
447     return (float4)(red, green, blue, alpha);
448   }
449 
450   static inline void WriteChannels(__global CLQuantum *p, const unsigned int number_channels,
451     const ChannelType channel, float red, float green, float blue, float alpha)
452   {
453     if ((channel & RedChannel) != 0)
454       setPixelRed(p,ClampToQuantum(red));
455 
456     if (number_channels > 2)
457       {
458         if ((channel & GreenChannel) != 0)
459           setPixelGreen(p,ClampToQuantum(green));
460 
461         if ((channel & BlueChannel) != 0)
462           setPixelBlue(p,ClampToQuantum(blue));
463       }
464 
465     if (((number_channels == 4) || (number_channels == 2)) &&
466         ((channel & AlphaChannel) != 0))
467       setPixelAlpha(p,number_channels,ClampToQuantum(alpha));
468   }
469 
470   static inline void WriteAllChannels(__global CLQuantum *image, const unsigned int number_channels,
471     const unsigned int columns, const unsigned int x, const unsigned int y, float4 pixel)
472   {
473     __global CLQuantum *p = image + getPixelIndex(number_channels, columns, x, y);
474 
475     setPixelRed(p,ClampToQuantum(pixel.x));
476 
477     if (number_channels > 2)
478       {
479         setPixelGreen(p,ClampToQuantum(pixel.y));
480         setPixelBlue(p,ClampToQuantum(pixel.z));
481       }
482 
483     if ((number_channels == 4) || (number_channels == 2))
484       setPixelAlpha(p,number_channels,ClampToQuantum(pixel.w));
485   }
486 
487   static inline void WriteFloat4(__global CLQuantum *image, const unsigned int number_channels,
488     const unsigned int columns, const unsigned int x, const unsigned int y, const ChannelType channel,
489     float4 pixel)
490   {
491     __global CLQuantum *p = image + getPixelIndex(number_channels, columns, x, y);
492     WriteChannels(p, number_channels, channel, pixel.x, pixel.y, pixel.z, pixel.w);
493   }
494 
495   static inline float GetPixelIntensity(const unsigned int colorspace,
496     const unsigned int method,float red,float green,float blue)
497   {
498     float intensity;
499 
500     if ((colorspace == GRAYColorspace) || (colorspace == LinearGRAYColorspace))
501       return red;
502 
503     switch (method)
504     {
505       case AveragePixelIntensityMethod:
506         {
507           intensity=(red+green+blue)/3.0;
508           break;
509         }
510       case BrightnessPixelIntensityMethod:
511         {
512           intensity=MagickMax(MagickMax(red,green),blue);
513           break;
514         }
515       case LightnessPixelIntensityMethod:
516         {
517           intensity=(MagickMin(MagickMin(red,green),blue)+
518               MagickMax(MagickMax(red,green),blue))/2.0;
519           break;
520         }
521       case MSPixelIntensityMethod:
522         {
523           intensity=(float) (((float) red*red+green*green+blue*blue)/
524               (3.0*QuantumRange));
525           break;
526         }
527       case Rec601LumaPixelIntensityMethod:
528         {
529           /*
530           if (image->colorspace == RGBColorspace)
531           {
532             red=EncodePixelGamma(red);
533             green=EncodePixelGamma(green);
534             blue=EncodePixelGamma(blue);
535           }
536           */
537           intensity=0.298839*red+0.586811*green+0.114350*blue;
538           break;
539         }
540       case Rec601LuminancePixelIntensityMethod:
541         {
542           /*
543           if (image->colorspace == sRGBColorspace)
544           {
545             red=DecodePixelGamma(red);
546             green=DecodePixelGamma(green);
547             blue=DecodePixelGamma(blue);
548           }
549           */
550           intensity=0.298839*red+0.586811*green+0.114350*blue;
551           break;
552         }
553       case Rec709LumaPixelIntensityMethod:
554       default:
555         {
556           /*
557           if (image->colorspace == RGBColorspace)
558           {
559             red=EncodePixelGamma(red);
560             green=EncodePixelGamma(green);
561             blue=EncodePixelGamma(blue);
562           }
563           */
564           intensity=0.212656*red+0.715158*green+0.072186*blue;
565           break;
566         }
567       case Rec709LuminancePixelIntensityMethod:
568         {
569           /*
570           if (image->colorspace == sRGBColorspace)
571           {
572             red=DecodePixelGamma(red);
573             green=DecodePixelGamma(green);
574             blue=DecodePixelGamma(blue);
575           }
576           */
577           intensity=0.212656*red+0.715158*green+0.072186*blue;
578           break;
579         }
580       case RMSPixelIntensityMethod:
581         {
582           intensity=(float) (sqrt((float) red*red+green*green+blue*blue)/
583               sqrt(3.0));
584           break;
585         }
586     }
587 
588     return intensity;
589   }
590 
591   static inline int mirrorBottom(int value)
592   {
593       return (value < 0) ? - (value) : value;
594   }
595 
596   static inline int mirrorTop(int value, int width)
597   {
598       return (value >= width) ? (2 * width - value - 1) : value;
599   }
600   )
601 
602 /*
603 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
604 %                                                                             %
605 %                                                                             %
606 %                                                                             %
607 %     A d d N o i s e                                                         %
608 %                                                                             %
609 %                                                                             %
610 %                                                                             %
611 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
612 */
613 
614   STRINGIFY(
615   /*
616   Part of MWC64X by David Thomas, dt10@imperial.ac.uk
617   This is provided under BSD, full license is with the main package.
618   See http://www.doc.ic.ac.uk/~dt10/research
619   */
620 
621   // Pre: a<M, b<M
622   // Post: r=(a+b) mod M
623   ulong MWC_AddMod64(ulong a, ulong b, ulong M)
624   {
625     ulong v=a+b;
626     //if( (v>=M) || (v<a) )
627     if( (v>=M) || (convert_float(v) < convert_float(a)) ) // workaround for what appears to be an optimizer bug.
628       v=v-M;
629     return v;
630   }
631 
632   // Pre: a<M,b<M
633   // Post: r=(a*b) mod M
634   // This could be done more efficently, but it is portable, and should
635   // be easy to understand. It can be replaced with any of the better
636   // modular multiplication algorithms (for example if you know you have
637   // double precision available or something).
638   ulong MWC_MulMod64(ulong a, ulong b, ulong M)
639   {
640     ulong r=0;
641     while(a!=0){
642       if(a&1)
643         r=MWC_AddMod64(r,b,M);
644       b=MWC_AddMod64(b,b,M);
645       a=a>>1;
646     }
647     return r;
648   }
649 
650   // Pre: a<M, e>=0
651   // Post: r=(a^b) mod M
652   // This takes at most ~64^2 modular additions, so probably about 2^15 or so instructions on
653   // most architectures
654   ulong MWC_PowMod64(ulong a, ulong e, ulong M)
655   {
656     ulong sqr=a, acc=1;
657     while(e!=0){
658       if(e&1)
659         acc=MWC_MulMod64(acc,sqr,M);
660         sqr=MWC_MulMod64(sqr,sqr,M);
661       e=e>>1;
662     }
663     return acc;
664   }
665 
666   uint2 MWC_SkipImpl_Mod64(uint2 curr, ulong A, ulong M, ulong distance)
667   {
668     ulong m=MWC_PowMod64(A, distance, M);
669     ulong x=curr.x*(ulong)A+curr.y;
670     x=MWC_MulMod64(x, m, M);
671     return (uint2)((uint)(x/A), (uint)(x%A));
672   }
673 
674   uint2 MWC_SeedImpl_Mod64(ulong A, ulong M, uint vecSize, uint vecOffset, ulong streamBase, ulong streamGap)
675   {
676     // This is an arbitrary constant for starting LCG jumping from. I didn't
677     // want to start from 1, as then you end up with the two or three first values
678     // being a bit poor in ones - once you've decided that, one constant is as
679     // good as any another. There is no deep mathematical reason for it, I just
680     // generated a random number.
681     enum{ MWC_BASEID = 4077358422479273989UL };
682 
683     ulong dist=streamBase + (get_global_id(0)*vecSize+vecOffset)*streamGap;
684     ulong m=MWC_PowMod64(A, dist, M);
685 
686     ulong x=MWC_MulMod64(MWC_BASEID, m, M);
687     return (uint2)((uint)(x/A), (uint)(x%A));
688   }
689 
690   //! Represents the state of a particular generator
691   typedef struct{ uint x; uint c; uint seed0; ulong seed1; } mwc64x_state_t;
692 
693   void MWC64X_Step(mwc64x_state_t *s)
694   {
695     uint X=s->x, C=s->c;
696 
697     uint Xn=s->seed0*X+C;
698     uint carry=(uint)(Xn<C); // The (Xn<C) will be zero or one for scalar
699     uint Cn=mad_hi(s->seed0,X,carry);
700 
701     s->x=Xn;
702     s->c=Cn;
703   }
704 
705   void MWC64X_Skip(mwc64x_state_t *s, ulong distance)
706   {
707     uint2 tmp=MWC_SkipImpl_Mod64((uint2)(s->x,s->c), s->seed0, s->seed1, distance);
708     s->x=tmp.x;
709     s->c=tmp.y;
710   }
711 
712   void MWC64X_SeedStreams(mwc64x_state_t *s, ulong baseOffset, ulong perStreamOffset)
713   {
714     uint2 tmp=MWC_SeedImpl_Mod64(s->seed0, s->seed1, 1, 0, baseOffset, perStreamOffset);
715     s->x=tmp.x;
716     s->c=tmp.y;
717   }
718 
719   //! Return a 32-bit integer in the range [0..2^32)
720   uint MWC64X_NextUint(mwc64x_state_t *s)
721   {
722     uint res=s->x ^ s->c;
723     MWC64X_Step(s);
724     return res;
725   }
726 
727   //
728   // End of MWC64X excerpt
729   //
730 
731   float mwcReadPseudoRandomValue(mwc64x_state_t* rng)
732   {
733     return (1.0f * MWC64X_NextUint(rng)) / (float)(0xffffffff); // normalized to 1.0
734   }
735 
736   float mwcGenerateDifferentialNoise(mwc64x_state_t* r, float pixel, NoiseType noise_type, float attenuate)
737   {
738     float
739       alpha,
740       beta,
741       noise,
742       sigma;
743 
744     noise = 0.0f;
745     alpha=mwcReadPseudoRandomValue(r);
746     switch(noise_type)
747     {
748       case UniformNoise:
749       default:
750         {
751           noise=(pixel+QuantumRange*SigmaUniform*(alpha-0.5f));
752           break;
753         }
754       case GaussianNoise:
755         {
756           float
757             gamma,
758             tau;
759 
760           if (alpha == 0.0f)
761             alpha=1.0f;
762           beta=mwcReadPseudoRandomValue(r);
763           gamma=sqrt(-2.0f*log(alpha));
764           sigma=gamma*cospi((2.0f*beta));
765           tau=gamma*sinpi((2.0f*beta));
766           noise=pixel+sqrt(pixel)*SigmaGaussian*sigma+QuantumRange*TauGaussian*tau;
767           break;
768         }
769       case ImpulseNoise:
770       {
771         if (alpha < (SigmaImpulse/2.0f))
772           noise=0.0f;
773         else
774           if (alpha >= (1.0f-(SigmaImpulse/2.0f)))
775             noise=QuantumRange;
776           else
777             noise=pixel;
778         break;
779       }
780       case LaplacianNoise:
781       {
782         if (alpha <= 0.5f)
783           {
784             if (alpha <= MagickEpsilon)
785               noise=(pixel-QuantumRange);
786             else
787               noise=(pixel+QuantumRange*SigmaLaplacian*log(2.0f*alpha)+
788                 0.5f);
789             break;
790           }
791         beta=1.0f-alpha;
792         if (beta <= (0.5f*MagickEpsilon))
793           noise=(pixel+QuantumRange);
794         else
795           noise=(pixel-QuantumRange*SigmaLaplacian*log(2.0f*beta)+0.5f);
796         break;
797       }
798       case MultiplicativeGaussianNoise:
799       {
800         sigma=1.0f;
801         if (alpha > MagickEpsilon)
802           sigma=sqrt(-2.0f*log(alpha));
803         beta=mwcReadPseudoRandomValue(r);
804         noise=(pixel+pixel*SigmaMultiplicativeGaussian*sigma*
805           cospi((2.0f*beta))/2.0f);
806         break;
807       }
808       case PoissonNoise:
809       {
810         float
811           poisson;
812         unsigned int i;
813         poisson=exp(-SigmaPoisson*QuantumScale*pixel);
814         for (i=0; alpha > poisson; i++)
815         {
816           beta=mwcReadPseudoRandomValue(r);
817           alpha*=beta;
818         }
819         noise=(QuantumRange*i/SigmaPoisson);
820         break;
821       }
822       case RandomNoise:
823       {
824         noise=(QuantumRange*SigmaRandom*alpha);
825         break;
826       }
827     }
828     return noise;
829   }
830 
831   __kernel
832   void AddNoise(const __global CLQuantum *image,
833     const unsigned int number_channels,const ChannelType channel,
834     const unsigned int length,const unsigned int pixelsPerWorkItem,
835     const NoiseType noise_type,const float attenuate,const unsigned int seed0,
836     const unsigned int seed1,const unsigned int numRandomNumbersPerPixel,
837     __global CLQuantum *filteredImage)
838   {
839     mwc64x_state_t rng;
840     rng.seed0 = seed0;
841     rng.seed1 = seed1;
842 
843     uint span = pixelsPerWorkItem * numRandomNumbersPerPixel; // length of RNG substream each workitem will use
844     uint offset = span * get_local_size(0) * get_group_id(0); // offset of this workgroup's RNG substream (in master stream);
845     MWC64X_SeedStreams(&rng, offset, span); // Seed the RNG streams
846 
847     uint pos = get_group_id(0) * get_local_size(0) * pixelsPerWorkItem * number_channels + (get_local_id(0) * number_channels);
848     uint count = pixelsPerWorkItem;
849 
850     while (count > 0 && pos < length)
851     {
852       const __global CLQuantum *p = image + pos;
853       __global CLQuantum *q = filteredImage + pos;
854 
855       float red;
856       float green;
857       float blue;
858       float alpha;
859 
860       ReadChannels(p, number_channels, channel, &red, &green, &blue, &alpha);
861 
862       if ((channel & RedChannel) != 0)
863         red=mwcGenerateDifferentialNoise(&rng,red,noise_type,attenuate);
864 
865       if (number_channels > 2)
866       {
867         if ((channel & GreenChannel) != 0)
868           green=mwcGenerateDifferentialNoise(&rng,green,noise_type,attenuate);
869 
870         if ((channel & BlueChannel) != 0)
871           blue=mwcGenerateDifferentialNoise(&rng,blue,noise_type,attenuate);
872       }
873 
874       if (((number_channels == 4) || (number_channels == 2)) &&
875           ((channel & AlphaChannel) != 0))
876         alpha=mwcGenerateDifferentialNoise(&rng,alpha,noise_type,attenuate);
877 
878       WriteChannels(q, number_channels, channel, red, green, blue, alpha);
879 
880       pos += (get_local_size(0) * number_channels);
881       count--;
882     }
883   }
884   )
885 
886 /*
887 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
888 %                                                                             %
889 %                                                                             %
890 %                                                                             %
891 %    B l u r                                                                  %
892 %                                                                             %
893 %                                                                             %
894 %                                                                             %
895 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
896 */
897 
898   STRINGIFY(
899   /*
900   Reduce image noise and reduce detail levels by row
901   */
902   __kernel void BlurRow(const __global CLQuantum *image,
903     const unsigned int number_channels,const ChannelType channel,
904     __constant float *filter,const unsigned int width,
905     const unsigned int imageColumns,const unsigned int imageRows,
906     __local float4 *temp,__global float4 *tempImage)
907   {
908     const int x = get_global_id(0);
909     const int y = get_global_id(1);
910 
911     const int columns = imageColumns;
912 
913     const unsigned int radius = (width-1)/2;
914     const int wsize = get_local_size(0);
915     const unsigned int loadSize = wsize+width;
916 
917     //group coordinate
918     const int groupX=get_local_size(0)*get_group_id(0);
919 
920     //parallel load and clamp
921     for (int i=get_local_id(0); i < loadSize; i=i+get_local_size(0))
922     {
923       int cx = ClampToCanvas(i + groupX - radius, columns);
924       temp[i] = ReadFloat4(image, number_channels, columns, cx, y, channel);
925     }
926 
927     // barrier
928     barrier(CLK_LOCAL_MEM_FENCE);
929 
930     // only do the work if this is not a patched item
931     if (get_global_id(0) < columns)
932     {
933       // compute
934       float4 result = (float4) 0;
935 
936       int i = 0;
937 
938       for ( ; i+7 < width; )
939       {
940         for (int j=0; j < 8; j++)
941           result+=filter[i+j]*temp[i+j+get_local_id(0)];
942         i+=8;
943       }
944 
945       for ( ; i < width; i++)
946         result+=filter[i]*temp[i+get_local_id(0)];
947 
948       // write back to global
949       tempImage[y*columns+x] = result;
950     }
951   }
952   )
953 
954   STRINGIFY(
955   /*
956   Reduce image noise and reduce detail levels by line
957   */
958   __kernel void BlurColumn(const __global float4 *blurRowData,
959     const unsigned int number_channels,const ChannelType channel,
960     __constant float *filter,const unsigned int width,
961     const unsigned int imageColumns,const unsigned int imageRows,
962     __local float4 *temp,__global CLQuantum *filteredImage)
963   {
964     const int x = get_global_id(0);
965     const int y = get_global_id(1);
966 
967     const int columns = imageColumns;
968     const int rows = imageRows;
969 
970     unsigned int radius = (width-1)/2;
971     const int wsize = get_local_size(1);
972     const unsigned int loadSize = wsize+width;
973 
974     //group coordinate
975     const int groupX=get_local_size(0)*get_group_id(0);
976     const int groupY=get_local_size(1)*get_group_id(1);
977     //notice that get_local_size(0) is 1, so
978     //groupX=get_group_id(0);
979 
980     //parallel load and clamp
981     for (int i = get_local_id(1); i < loadSize; i=i+get_local_size(1))
982       temp[i] = blurRowData[ClampToCanvas(i+groupY-radius, rows) * columns + groupX];
983 
984     // barrier
985     barrier(CLK_LOCAL_MEM_FENCE);
986 
987     // only do the work if this is not a patched item
988     if (get_global_id(1) < rows)
989     {
990       // compute
991       float4 result = (float4) 0;
992 
993       int i = 0;
994 
995       for ( ; i+7 < width; )
996       {
997         for (int j=0; j < 8; j++)
998           result+=filter[i+j]*temp[i+j+get_local_id(1)];
999         i+=8;
1000       }
1001 
1002       for ( ; i < width; i++)
1003         result+=filter[i]*temp[i+get_local_id(1)];
1004 
1005       // write back to global
1006       WriteFloat4(filteredImage, number_channels, columns, x, y, channel, result);
1007     }
1008   }
1009   )
1010 
1011 /*
1012 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1013 %                                                                             %
1014 %                                                                             %
1015 %                                                                             %
1016 %    C o n t r a s t                                                          %
1017 %                                                                             %
1018 %                                                                             %
1019 %                                                                             %
1020 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1021 */
1022 
1023   STRINGIFY(
1024 
1025   static inline float4 ConvertRGBToHSB(const float4 pixel)
1026   {
1027     float4 result=0.0f;
1028     result.w=pixel.w;
1029     float tmax=MagickMax(MagickMax(pixel.x,pixel.y),pixel.z);
1030     if (tmax != 0.0f)
1031     {
1032       float tmin=MagickMin(MagickMin(pixel.x,pixel.y),pixel.z);
1033       float delta=tmax-tmin;
1034 
1035       result.y=delta/tmax;
1036       result.z=QuantumScale*tmax;
1037       if (delta != 0.0f)
1038       {
1039         result.x =((pixel.x == tmax) ? 0.0f : ((pixel.y == tmax) ? 2.0f : 4.0f));
1040         result.x+=((pixel.x == tmax) ? (pixel.y-pixel.z) : ((pixel.y == tmax) ?
1041           (pixel.z-pixel.x) : (pixel.x-pixel.y)))/delta;
1042         result.x/=6.0f;
1043         result.x+=(result.x < 0.0f) ? 0.0f : 1.0f;
1044       }
1045     }
1046     return(result);
1047   }
1048 
1049   static inline float4 ConvertHSBToRGB(const float4 pixel)
1050   {
1051     float hue=pixel.x;
1052     float saturation=pixel.y;
1053     float brightness=pixel.z;
1054 
1055     float4 result=pixel;
1056 
1057     if (saturation == 0.0f)
1058     {
1059       result.x=result.y=result.z=ClampToQuantum(QuantumRange*brightness);
1060     }
1061     else
1062     {
1063       float h=6.0f*(hue-floor(hue));
1064       float f=h-floor(h);
1065       float p=brightness*(1.0f-saturation);
1066       float q=brightness*(1.0f-saturation*f);
1067       float t=brightness*(1.0f-(saturation*(1.0f-f)));
1068       int ih = (int)h;
1069 
1070       if (ih == 1)
1071       {
1072         result.x=ClampToQuantum(QuantumRange*q);
1073         result.y=ClampToQuantum(QuantumRange*brightness);
1074         result.z=ClampToQuantum(QuantumRange*p);
1075       }
1076       else if (ih == 2)
1077       {
1078         result.x=ClampToQuantum(QuantumRange*p);
1079         result.y=ClampToQuantum(QuantumRange*brightness);
1080         result.z=ClampToQuantum(QuantumRange*t);
1081       }
1082       else if (ih == 3)
1083       {
1084         result.x=ClampToQuantum(QuantumRange*p);
1085         result.y=ClampToQuantum(QuantumRange*q);
1086         result.z=ClampToQuantum(QuantumRange*brightness);
1087       }
1088       else if (ih == 4)
1089       {
1090         result.x=ClampToQuantum(QuantumRange*t);
1091         result.y=ClampToQuantum(QuantumRange*p);
1092         result.z=ClampToQuantum(QuantumRange*brightness);
1093       }
1094       else if (ih == 5)
1095       {
1096         result.x=ClampToQuantum(QuantumRange*brightness);
1097         result.y=ClampToQuantum(QuantumRange*p);
1098         result.z=ClampToQuantum(QuantumRange*q);
1099       }
1100       else
1101       {
1102         result.x=ClampToQuantum(QuantumRange*brightness);
1103         result.y=ClampToQuantum(QuantumRange*t);
1104         result.z=ClampToQuantum(QuantumRange*p);
1105       }
1106     }
1107     return(result);
1108   }
1109 
1110   __kernel void Contrast(__global CLQuantum *image,
1111     const unsigned int number_channels,const int sign)
1112   {
1113     const int x=get_global_id(0);
1114     const int y=get_global_id(1);
1115     const unsigned int columns=get_global_size(0);
1116 
1117     float4 pixel=ReadAllChannels(image,number_channels,columns,x,y);
1118     if (number_channels < 3)
1119       pixel.y=pixel.z=pixel.x;
1120 
1121     pixel=ConvertRGBToHSB(pixel);
1122     float brightness=pixel.z;
1123     brightness+=0.5f*sign*(0.5f*(sinpi(brightness-0.5f)+1.0f)-brightness);
1124     brightness=clamp(brightness,0.0f,1.0f);
1125     pixel.z=brightness;
1126     pixel=ConvertHSBToRGB(pixel);
1127 
1128     WriteAllChannels(image,number_channels,columns,x,y,pixel);
1129   }
1130   )
1131 
1132 /*
1133 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1134 %                                                                             %
1135 %                                                                             %
1136 %                                                                             %
1137 %    C o n t r a s t S t r e t c h                                            %
1138 %                                                                             %
1139 %                                                                             %
1140 %                                                                             %
1141 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1142 */
1143 
1144     STRINGIFY(
1145     /*
1146     */
1147     __kernel void Histogram(__global CLPixelType * restrict im,
1148       const ChannelType channel,
1149       const unsigned int colorspace,
1150       const unsigned int method,
1151       __global uint4 * restrict histogram)
1152       {
1153         const int x = get_global_id(0);
1154         const int y = get_global_id(1);
1155         const int columns = get_global_size(0);
1156         const int c = x + y * columns;
1157         if ((channel & SyncChannels) != 0)
1158         {
1159           float red=(float)getRed(im[c]);
1160           float green=(float)getGreen(im[c]);
1161           float blue=(float)getBlue(im[c]);
1162 
1163           float intensity = GetPixelIntensity(colorspace, method, red, green, blue);
1164           uint pos = ScaleQuantumToMap(ClampToQuantum(intensity));
1165           atomic_inc((__global uint *)(&(histogram[pos]))+2); //red position
1166         }
1167         else
1168         {
1169           // for equalizing, we always need all channels?
1170           // otherwise something more
1171         }
1172       }
1173     )
1174 
1175     STRINGIFY(
1176     /*
1177     */
1178     __kernel void ContrastStretch(__global CLPixelType * restrict im,
1179       const ChannelType channel,
1180       __global CLPixelType * restrict stretch_map,
1181       const float4 white, const float4 black)
1182       {
1183         const int x = get_global_id(0);
1184         const int y = get_global_id(1);
1185         const int columns = get_global_size(0);
1186         const int c = x + y * columns;
1187 
1188         uint ePos;
1189         CLPixelType oValue, eValue;
1190         CLQuantum red, green, blue, alpha;
1191 
1192         //read from global
1193         oValue=im[c];
1194 
1195         if ((channel & RedChannel) != 0)
1196         {
1197           if (getRedF4(white) != getRedF4(black))
1198           {
1199             ePos = ScaleQuantumToMap(getRed(oValue));
1200             eValue = stretch_map[ePos];
1201             red = getRed(eValue);
1202           }
1203         }
1204 
1205         if ((channel & GreenChannel) != 0)
1206         {
1207           if (getGreenF4(white) != getGreenF4(black))
1208           {
1209             ePos = ScaleQuantumToMap(getGreen(oValue));
1210             eValue = stretch_map[ePos];
1211             green = getGreen(eValue);
1212           }
1213         }
1214 
1215         if ((channel & BlueChannel) != 0)
1216         {
1217           if (getBlueF4(white) != getBlueF4(black))
1218           {
1219             ePos = ScaleQuantumToMap(getBlue(oValue));
1220             eValue = stretch_map[ePos];
1221             blue = getBlue(eValue);
1222           }
1223         }
1224 
1225         if ((channel & AlphaChannel) != 0)
1226         {
1227           if (getAlphaF4(white) != getAlphaF4(black))
1228           {
1229             ePos = ScaleQuantumToMap(getAlpha(oValue));
1230             eValue = stretch_map[ePos];
1231             alpha = getAlpha(eValue);
1232           }
1233         }
1234 
1235         //write back
1236         im[c]=(CLPixelType)(blue, green, red, alpha);
1237 
1238       }
1239     )
1240 
1241 /*
1242 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1243 %                                                                             %
1244 %                                                                             %
1245 %                                                                             %
1246 %    C o n v o l v e                                                          %
1247 %                                                                             %
1248 %                                                                             %
1249 %                                                                             %
1250 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1251 */
1252 
1253   STRINGIFY(
1254     __kernel
1255     void ConvolveOptimized(const __global CLPixelType *input, __global CLPixelType *output,
1256     const unsigned int imageWidth, const unsigned int imageHeight,
1257     __constant float *filter, const unsigned int filterWidth, const unsigned int filterHeight,
1258     const uint matte, const ChannelType channel, __local CLPixelType *pixelLocalCache, __local float* filterCache) {
1259 
1260       int2 blockID;
1261       blockID.x = get_global_id(0) / get_local_size(0);
1262       blockID.y = get_global_id(1) / get_local_size(1);
1263 
1264       // image area processed by this workgroup
1265       int2 imageAreaOrg;
1266       imageAreaOrg.x = blockID.x * get_local_size(0);
1267       imageAreaOrg.y = blockID.y * get_local_size(1);
1268 
1269       int2 midFilterDimen;
1270       midFilterDimen.x = (filterWidth-1)/2;
1271       midFilterDimen.y = (filterHeight-1)/2;
1272 
1273       int2 cachedAreaOrg = imageAreaOrg - midFilterDimen;
1274 
1275       // dimension of the local cache
1276       int2 cachedAreaDimen;
1277       cachedAreaDimen.x = get_local_size(0) + filterWidth - 1;
1278       cachedAreaDimen.y = get_local_size(1) + filterHeight - 1;
1279 
1280       // cache the pixels accessed by this workgroup in local memory
1281       int localID = get_local_id(1)*get_local_size(0)+get_local_id(0);
1282       int cachedAreaNumPixels = cachedAreaDimen.x * cachedAreaDimen.y;
1283       int groupSize = get_local_size(0) * get_local_size(1);
1284       for (int i = localID; i < cachedAreaNumPixels; i+=groupSize) {
1285 
1286         int2 cachedAreaIndex;
1287         cachedAreaIndex.x = i % cachedAreaDimen.x;
1288         cachedAreaIndex.y = i / cachedAreaDimen.x;
1289 
1290         int2 imagePixelIndex;
1291         imagePixelIndex = cachedAreaOrg + cachedAreaIndex;
1292 
1293         // only support EdgeVirtualPixelMethod through ClampToCanvas
1294         // TODO: implement other virtual pixel method
1295         imagePixelIndex.x = ClampToCanvas(imagePixelIndex.x, imageWidth);
1296         imagePixelIndex.y = ClampToCanvas(imagePixelIndex.y, imageHeight);
1297 
1298         pixelLocalCache[i] = input[imagePixelIndex.y * imageWidth + imagePixelIndex.x];
1299       }
1300 
1301       // cache the filter
1302       for (int i = localID; i < filterHeight*filterWidth; i+=groupSize) {
1303         filterCache[i] = filter[i];
1304       }
1305       barrier(CLK_LOCAL_MEM_FENCE);
1306 
1307 
1308       int2 imageIndex;
1309       imageIndex.x = imageAreaOrg.x + get_local_id(0);
1310       imageIndex.y = imageAreaOrg.y + get_local_id(1);
1311 
1312       // if out-of-range, stops here and quit
1313       if (imageIndex.x >= imageWidth
1314         || imageIndex.y >= imageHeight) {
1315           return;
1316       }
1317 
1318       int filterIndex = 0;
1319       float4 sum = (float4)0.0f;
1320       float gamma = 0.0f;
1321       if (((channel & AlphaChannel) == 0) || (matte == 0)) {
1322         int cacheIndexY = get_local_id(1);
1323         for (int j = 0; j < filterHeight; j++) {
1324           int cacheIndexX = get_local_id(0);
1325           for (int i = 0; i < filterWidth; i++) {
1326             CLPixelType p = pixelLocalCache[cacheIndexY*cachedAreaDimen.x + cacheIndexX];
1327             float f = filterCache[filterIndex];
1328 
1329             sum.x += f * p.x;
1330             sum.y += f * p.y;
1331             sum.z += f * p.z;
1332             sum.w += f * p.w;
1333 
1334             gamma += f;
1335             filterIndex++;
1336             cacheIndexX++;
1337           }
1338           cacheIndexY++;
1339         }
1340       }
1341       else {
1342         int cacheIndexY = get_local_id(1);
1343         for (int j = 0; j < filterHeight; j++) {
1344           int cacheIndexX = get_local_id(0);
1345           for (int i = 0; i < filterWidth; i++) {
1346 
1347             CLPixelType p = pixelLocalCache[cacheIndexY*cachedAreaDimen.x + cacheIndexX];
1348             float alpha = QuantumScale*p.w;
1349             float f = filterCache[filterIndex];
1350             float g = alpha * f;
1351 
1352             sum.x += g*p.x;
1353             sum.y += g*p.y;
1354             sum.z += g*p.z;
1355             sum.w += f*p.w;
1356 
1357             gamma += g;
1358             filterIndex++;
1359             cacheIndexX++;
1360           }
1361           cacheIndexY++;
1362         }
1363         gamma = PerceptibleReciprocal(gamma);
1364         sum.xyz = gamma*sum.xyz;
1365       }
1366       CLPixelType outputPixel;
1367       outputPixel.x = ClampToQuantum(sum.x);
1368       outputPixel.y = ClampToQuantum(sum.y);
1369       outputPixel.z = ClampToQuantum(sum.z);
1370       outputPixel.w = ((channel & AlphaChannel)!=0)?ClampToQuantum(sum.w):input[imageIndex.y * imageWidth + imageIndex.x].w;
1371 
1372       output[imageIndex.y * imageWidth + imageIndex.x] = outputPixel;
1373     }
1374   )
1375 
1376   STRINGIFY(
1377     __kernel
1378     void Convolve(const __global CLPixelType *input, __global CLPixelType *output,
1379                   const uint imageWidth, const uint imageHeight,
1380                   __constant float *filter, const unsigned int filterWidth, const unsigned int filterHeight,
1381                   const uint matte, const ChannelType channel) {
1382 
1383       int2 imageIndex;
1384       imageIndex.x = get_global_id(0);
1385       imageIndex.y = get_global_id(1);
1386 
1387       /*
1388       unsigned int imageWidth = get_global_size(0);
1389       unsigned int imageHeight = get_global_size(1);
1390       */
1391       if (imageIndex.x >= imageWidth
1392           || imageIndex.y >= imageHeight)
1393           return;
1394 
1395       int2 midFilterDimen;
1396       midFilterDimen.x = (filterWidth-1)/2;
1397       midFilterDimen.y = (filterHeight-1)/2;
1398 
1399       int filterIndex = 0;
1400       float4 sum = (float4)0.0f;
1401       float gamma = 0.0f;
1402       if (((channel & AlphaChannel) == 0) || (matte == 0)) {
1403         for (int j = 0; j < filterHeight; j++) {
1404           int2 inputPixelIndex;
1405           inputPixelIndex.y = imageIndex.y - midFilterDimen.y + j;
1406           inputPixelIndex.y = ClampToCanvas(inputPixelIndex.y, imageHeight);
1407           for (int i = 0; i < filterWidth; i++) {
1408             inputPixelIndex.x = imageIndex.x - midFilterDimen.x + i;
1409             inputPixelIndex.x = ClampToCanvas(inputPixelIndex.x, imageWidth);
1410 
1411             CLPixelType p = input[inputPixelIndex.y * imageWidth + inputPixelIndex.x];
1412             float f = filter[filterIndex];
1413 
1414             sum.x += f * p.x;
1415             sum.y += f * p.y;
1416             sum.z += f * p.z;
1417             sum.w += f * p.w;
1418 
1419             gamma += f;
1420 
1421             filterIndex++;
1422           }
1423         }
1424       }
1425       else {
1426 
1427         for (int j = 0; j < filterHeight; j++) {
1428           int2 inputPixelIndex;
1429           inputPixelIndex.y = imageIndex.y - midFilterDimen.y + j;
1430           inputPixelIndex.y = ClampToCanvas(inputPixelIndex.y, imageHeight);
1431           for (int i = 0; i < filterWidth; i++) {
1432             inputPixelIndex.x = imageIndex.x - midFilterDimen.x + i;
1433             inputPixelIndex.x = ClampToCanvas(inputPixelIndex.x, imageWidth);
1434 
1435             CLPixelType p = input[inputPixelIndex.y * imageWidth + inputPixelIndex.x];
1436             float alpha = QuantumScale*p.w;
1437             float f = filter[filterIndex];
1438             float g = alpha * f;
1439 
1440             sum.x += g*p.x;
1441             sum.y += g*p.y;
1442             sum.z += g*p.z;
1443             sum.w += f*p.w;
1444 
1445             gamma += g;
1446 
1447 
1448             filterIndex++;
1449           }
1450         }
1451         gamma = PerceptibleReciprocal(gamma);
1452         sum.xyz = gamma*sum.xyz;
1453       }
1454 
1455       CLPixelType outputPixel;
1456       outputPixel.x = ClampToQuantum(sum.x);
1457       outputPixel.y = ClampToQuantum(sum.y);
1458       outputPixel.z = ClampToQuantum(sum.z);
1459       outputPixel.w = ((channel & AlphaChannel)!=0)?ClampToQuantum(sum.w):input[imageIndex.y * imageWidth + imageIndex.x].w;
1460 
1461       output[imageIndex.y * imageWidth + imageIndex.x] = outputPixel;
1462     }
1463   )
1464 
1465 /*
1466 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1467 %                                                                             %
1468 %                                                                             %
1469 %                                                                             %
1470 %     D e s p e c k l e                                                       %
1471 %                                                                             %
1472 %                                                                             %
1473 %                                                                             %
1474 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1475 */
1476 
1477   STRINGIFY(
1478 
1479   __kernel void HullPass1(const __global CLPixelType *inputImage, __global CLPixelType *outputImage
1480   , const unsigned int imageWidth, const unsigned int imageHeight
1481   , const int2 offset, const int polarity, const int matte) {
1482 
1483     int x = get_global_id(0);
1484     int y = get_global_id(1);
1485 
1486     CLPixelType v = inputImage[y*imageWidth+x];
1487 
1488     int2 neighbor;
1489     neighbor.y = y + offset.y;
1490     neighbor.x = x + offset.x;
1491 
1492     int2 clampedNeighbor;
1493     clampedNeighbor.x = ClampToCanvas(neighbor.x, imageWidth);
1494     clampedNeighbor.y = ClampToCanvas(neighbor.y, imageHeight);
1495 
1496     CLPixelType r = (clampedNeighbor.x == neighbor.x
1497                      && clampedNeighbor.y == neighbor.y)?inputImage[clampedNeighbor.y*imageWidth+clampedNeighbor.x]
1498     :(CLPixelType)0;
1499 
1500     int sv[4];
1501     sv[0] = (int)v.x;
1502     sv[1] = (int)v.y;
1503     sv[2] = (int)v.z;
1504     sv[3] = (int)v.w;
1505 
1506     int sr[4];
1507     sr[0] = (int)r.x;
1508     sr[1] = (int)r.y;
1509     sr[2] = (int)r.z;
1510     sr[3] = (int)r.w;
1511 
1512     if (polarity > 0) {
1513       \n #pragma unroll 4\n
1514       for (unsigned int i = 0; i < 4; i++) {
1515         sv[i] = (sr[i] >= (sv[i]+ScaleCharToQuantum(2)))?(sv[i]+ScaleCharToQuantum(1)):sv[i];
1516       }
1517     }
1518     else {
1519       \n #pragma unroll 4\n
1520       for (unsigned int i = 0; i < 4; i++) {
1521         sv[i] = (sr[i] <= (sv[i]-ScaleCharToQuantum(2)))?(sv[i]-ScaleCharToQuantum(1)):sv[i];
1522       }
1523 
1524     }
1525 
1526     v.x = (CLQuantum)sv[0];
1527     v.y = (CLQuantum)sv[1];
1528     v.z = (CLQuantum)sv[2];
1529 
1530     if (matte!=0)
1531       v.w = (CLQuantum)sv[3];
1532 
1533     outputImage[y*imageWidth+x] = v;
1534 
1535     }
1536 
1537 
1538   )
1539 
1540   STRINGIFY(
1541 
1542   __kernel void HullPass2(const __global CLPixelType *inputImage, __global CLPixelType *outputImage
1543   , const unsigned int imageWidth, const unsigned int imageHeight
1544   , const int2 offset, const int polarity, const int matte) {
1545 
1546     int x = get_global_id(0);
1547     int y = get_global_id(1);
1548 
1549     CLPixelType v = inputImage[y*imageWidth+x];
1550 
1551     int2 neighbor, clampedNeighbor;
1552 
1553     neighbor.y = y + offset.y;
1554     neighbor.x = x + offset.x;
1555     clampedNeighbor.x = ClampToCanvas(neighbor.x, imageWidth);
1556     clampedNeighbor.y = ClampToCanvas(neighbor.y, imageHeight);
1557 
1558     CLPixelType r = (clampedNeighbor.x == neighbor.x
1559       && clampedNeighbor.y == neighbor.y)?inputImage[clampedNeighbor.y*imageWidth+clampedNeighbor.x]
1560     :(CLPixelType)0;
1561 
1562 
1563     neighbor.y = y - offset.y;
1564     neighbor.x = x - offset.x;
1565     clampedNeighbor.x = ClampToCanvas(neighbor.x, imageWidth);
1566     clampedNeighbor.y = ClampToCanvas(neighbor.y, imageHeight);
1567 
1568     CLPixelType s = (clampedNeighbor.x == neighbor.x
1569       && clampedNeighbor.y == neighbor.y)?inputImage[clampedNeighbor.y*imageWidth+clampedNeighbor.x]
1570     :(CLPixelType)0;
1571 
1572 
1573     int sv[4];
1574     sv[0] = (int)v.x;
1575     sv[1] = (int)v.y;
1576     sv[2] = (int)v.z;
1577     sv[3] = (int)v.w;
1578 
1579     int sr[4];
1580     sr[0] = (int)r.x;
1581     sr[1] = (int)r.y;
1582     sr[2] = (int)r.z;
1583     sr[3] = (int)r.w;
1584 
1585     int ss[4];
1586     ss[0] = (int)s.x;
1587     ss[1] = (int)s.y;
1588     ss[2] = (int)s.z;
1589     ss[3] = (int)s.w;
1590 
1591     if (polarity > 0) {
1592       \n #pragma unroll 4\n
1593       for (unsigned int i = 0; i < 4; i++) {
1594         //sv[i] = (ss[i] >= (sv[i]+ScaleCharToQuantum(2)) && sr[i] > sv[i] )   ? (sv[i]+ScaleCharToQuantum(1)):sv[i];
1595         //
1596         //sv[i] =(!( (int)(ss[i] >= (sv[i]+ScaleCharToQuantum(2))) && (int) (sr[i] > sv[i] ) ))  ? sv[i]:(sv[i]+ScaleCharToQuantum(1));
1597         //sv[i] =(( (int)( ss[i] < (sv[i]+ScaleCharToQuantum(2))) || (int) ( sr[i] <= sv[i] ) ))  ? sv[i]:(sv[i]+ScaleCharToQuantum(1));
1598         sv[i] =(( (int)( ss[i] < (sv[i]+ScaleCharToQuantum(2))) + (int) ( sr[i] <= sv[i] ) ) !=0)  ? sv[i]:(sv[i]+ScaleCharToQuantum(1));
1599       }
1600     }
1601     else {
1602       \n #pragma unroll 4\n
1603       for (unsigned int i = 0; i < 4; i++) {
1604         //sv[i] = (ss[i] <= (sv[i]-ScaleCharToQuantum(2)) && sr[i] < sv[i] )   ? (sv[i]-ScaleCharToQuantum(1)):sv[i];
1605         //
1606         //sv[i] = ( (int)(ss[i] <= (sv[i]-ScaleCharToQuantum(2)) ) + (int)( sr[i] < sv[i] ) ==0)   ? sv[i]:(sv[i]-ScaleCharToQuantum(1));
1607         sv[i] = (( (int)(ss[i] > (sv[i]-ScaleCharToQuantum(2))) + (int)( sr[i] >= sv[i] )) !=0)   ? sv[i]:(sv[i]-ScaleCharToQuantum(1));
1608       }
1609     }
1610 
1611     v.x = (CLQuantum)sv[0];
1612     v.y = (CLQuantum)sv[1];
1613     v.z = (CLQuantum)sv[2];
1614 
1615     if (matte!=0)
1616       v.w = (CLQuantum)sv[3];
1617 
1618     outputImage[y*imageWidth+x] = v;
1619 
1620     }
1621   )
1622 
1623 /*
1624 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1625 %                                                                             %
1626 %                                                                             %
1627 %                                                                             %
1628 %     E q u a l i z e                                                         %
1629 %                                                                             %
1630 %                                                                             %
1631 %                                                                             %
1632 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1633 */
1634 
1635     STRINGIFY(
1636     /*
1637     */
1638     __kernel void Equalize(__global CLPixelType * restrict im,
1639       const ChannelType channel,
1640       __global CLPixelType * restrict equalize_map,
1641       const float4 white, const float4 black)
1642       {
1643         const int x = get_global_id(0);
1644         const int y = get_global_id(1);
1645         const int columns = get_global_size(0);
1646         const int c = x + y * columns;
1647 
1648         uint ePos;
1649         CLPixelType oValue, eValue;
1650         CLQuantum red, green, blue, alpha;
1651 
1652         //read from global
1653         oValue=im[c];
1654 
1655         if ((channel & SyncChannels) != 0)
1656         {
1657           if (getRedF4(white) != getRedF4(black))
1658           {
1659             ePos = ScaleQuantumToMap(getRed(oValue));
1660             eValue = equalize_map[ePos];
1661             red = getRed(eValue);
1662             ePos = ScaleQuantumToMap(getGreen(oValue));
1663             eValue = equalize_map[ePos];
1664             green = getRed(eValue);
1665             ePos = ScaleQuantumToMap(getBlue(oValue));
1666             eValue = equalize_map[ePos];
1667             blue = getRed(eValue);
1668             ePos = ScaleQuantumToMap(getAlpha(oValue));
1669             eValue = equalize_map[ePos];
1670             alpha = getRed(eValue);
1671 
1672             //write back
1673             im[c]=(CLPixelType)(blue, green, red, alpha);
1674           }
1675 
1676         }
1677 
1678         // for equalizing, we always need all channels?
1679         // otherwise something more
1680 
1681      }
1682     )
1683 
1684 /*
1685 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1686 %                                                                             %
1687 %                                                                             %
1688 %                                                                             %
1689 %     F u n c t i o n                                                         %
1690 %                                                                             %
1691 %                                                                             %
1692 %                                                                             %
1693 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1694 */
1695 
1696   STRINGIFY(
1697   /*
1698   apply FunctionImageChannel(braightness-contrast)
1699   */
1700   CLQuantum ApplyFunction(float pixel,const MagickFunction function,
1701     const unsigned int number_parameters,__constant float *parameters)
1702   {
1703     float result = 0.0f;
1704 
1705     switch (function)
1706     {
1707     case PolynomialFunction:
1708       {
1709         for (unsigned int i=0; i < number_parameters; i++)
1710           result = result*QuantumScale*pixel + parameters[i];
1711         result *= QuantumRange;
1712         break;
1713       }
1714     case SinusoidFunction:
1715       {
1716         float  freq,phase,ampl,bias;
1717         freq  = ( number_parameters >= 1 ) ? parameters[0] : 1.0f;
1718         phase = ( number_parameters >= 2 ) ? parameters[1] : 0.0f;
1719         ampl  = ( number_parameters >= 3 ) ? parameters[2] : 0.5f;
1720         bias  = ( number_parameters >= 4 ) ? parameters[3] : 0.5f;
1721         result = QuantumRange*(ampl*sin(2.0f*MagickPI*
1722           (freq*QuantumScale*pixel + phase/360.0f)) + bias);
1723         break;
1724       }
1725     case ArcsinFunction:
1726       {
1727         float  width,range,center,bias;
1728         width  = ( number_parameters >= 1 ) ? parameters[0] : 1.0f;
1729         center = ( number_parameters >= 2 ) ? parameters[1] : 0.5f;
1730         range  = ( number_parameters >= 3 ) ? parameters[2] : 1.0f;
1731         bias   = ( number_parameters >= 4 ) ? parameters[3] : 0.5f;
1732 
1733         result = 2.0f/width*(QuantumScale*pixel - center);
1734         result = range/MagickPI*asin(result)+bias;
1735         result = ( result <= -1.0f ) ? bias - range/2.0f : result;
1736         result = ( result >= 1.0f ) ? bias + range/2.0f : result;
1737         result *= QuantumRange;
1738         break;
1739       }
1740     case ArctanFunction:
1741       {
1742         float slope,range,center,bias;
1743         slope  = ( number_parameters >= 1 ) ? parameters[0] : 1.0f;
1744         center = ( number_parameters >= 2 ) ? parameters[1] : 0.5f;
1745         range  = ( number_parameters >= 3 ) ? parameters[2] : 1.0f;
1746         bias   = ( number_parameters >= 4 ) ? parameters[3] : 0.5f;
1747         result = MagickPI*slope*(QuantumScale*pixel-center);
1748         result = QuantumRange*(range/MagickPI*atan(result) + bias);
1749         break;
1750       }
1751     case UndefinedFunction:
1752       break;
1753     }
1754     return(ClampToQuantum(result));
1755   }
1756   )
1757 
1758   STRINGIFY(
1759   /*
1760   Improve brightness / contrast of the image
1761   channel : define which channel is improved
1762   function : the function called to enchance the brightness contrast
1763   number_parameters : numbers of parameters
1764   parameters : the parameter
1765   */
1766   __kernel void ComputeFunction(__global CLQuantum *image,const unsigned int number_channels,
1767     const ChannelType channel,const MagickFunction function,const unsigned int number_parameters,
1768     __constant float *parameters)
1769   {
1770     const unsigned int x = get_global_id(0);
1771     const unsigned int y = get_global_id(1);
1772     const unsigned int columns = get_global_size(0);
1773     __global CLQuantum *p = image + getPixelIndex(number_channels, columns, x, y);
1774 
1775     float red;
1776     float green;
1777     float blue;
1778     float alpha;
1779 
1780     ReadChannels(p, number_channels, channel, &red, &green, &blue, &alpha);
1781 
1782     if ((channel & RedChannel) != 0)
1783       red=ApplyFunction(red, function, number_parameters, parameters);
1784 
1785     if (number_channels > 2)
1786       {
1787         if ((channel & GreenChannel) != 0)
1788           green=ApplyFunction(green, function, number_parameters, parameters);
1789 
1790         if ((channel & BlueChannel) != 0)
1791           blue=ApplyFunction(blue, function, number_parameters, parameters);
1792       }
1793 
1794     if (((number_channels == 4) || (number_channels == 2)) &&
1795         ((channel & AlphaChannel) != 0))
1796       alpha=ApplyFunction(alpha, function, number_parameters, parameters);
1797 
1798     WriteChannels(p, number_channels, channel, red, green, blue, alpha);
1799   }
1800   )
1801 
1802 /*
1803 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1804 %                                                                             %
1805 %                                                                             %
1806 %                                                                             %
1807 %     G r a y s c a l e                                                       %
1808 %                                                                             %
1809 %                                                                             %
1810 %                                                                             %
1811 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1812 */
1813 
1814   STRINGIFY(
1815   __kernel void Grayscale(__global CLQuantum *image,const int number_channels,
1816     const unsigned int colorspace,const unsigned int method)
1817   {
1818     const unsigned int x = get_global_id(0);
1819     const unsigned int y = get_global_id(1);
1820     const unsigned int columns = get_global_size(0);
1821     __global CLQuantum *p = image + getPixelIndex(number_channels, columns, x, y);
1822 
1823     float
1824       blue,
1825       green,
1826       red;
1827 
1828     red=getPixelRed(p);
1829     green=getPixelGreen(p);
1830     blue=getPixelBlue(p);
1831 
1832     CLQuantum intensity=ClampToQuantum(GetPixelIntensity(colorspace, method, red, green, blue));
1833 
1834     setPixelRed(p,intensity);
1835     setPixelGreen(p,intensity);
1836     setPixelBlue(p,intensity);
1837   }
1838   )
1839 
1840 /*
1841 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1842 %                                                                             %
1843 %                                                                             %
1844 %                                                                             %
1845 %     L o c a l C o n t r a s t                                               %
1846 %                                                                             %
1847 %                                                                             %
1848 %                                                                             %
1849 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1850 */
1851 
1852     STRINGIFY(
1853 
1854       __kernel void LocalContrastBlurRow(__global CLPixelType *srcImage, __global CLPixelType *dstImage, __global float *tmpImage,
1855           const int radius,
1856           const int imageWidth,
1857           const int imageHeight)
1858       {
1859         const float4 RGB = ((float4)(0.2126f, 0.7152f, 0.0722f, 0.0f));
1860 
1861         int x = get_local_id(0);
1862         int y = get_global_id(1);
1863 
1864         if ((x >= imageWidth) || (y >= imageHeight))
1865           return;
1866 
1867         global CLPixelType *src = srcImage + y * imageWidth;
1868 
1869         for (int i = x; i < imageWidth; i += get_local_size(0)) {
1870             float sum = 0.0f;
1871             float weight = 1.0f;
1872 
1873             int j = i - radius;
1874             while ((j + 7) < i) {
1875                 for (int k = 0; k < 8; ++k) // Unroll 8x
1876                     sum += (weight + k) * dot(RGB, convert_float4(src[mirrorBottom(j+k)]));
1877                 weight += 8.0f;
1878                 j+=8;
1879             }
1880             while (j < i) {
1881                 sum += weight * dot(RGB, convert_float4(src[mirrorBottom(j)]));
1882                 weight += 1.0f;
1883                 ++j;
1884             }
1885 
1886             while ((j + 7) < radius + i) {
1887                 for (int k = 0; k < 8; ++k) // Unroll 8x
1888                     sum += (weight - k) * dot(RGB, convert_float4(src[mirrorTop(j + k, imageWidth)]));
1889                 weight -= 8.0f;
1890                 j+=8;
1891             }
1892             while (j < radius + i) {
1893                 sum += weight * dot(RGB, convert_float4(src[mirrorTop(j, imageWidth)]));
1894                 weight -= 1.0f;
1895                 ++j;
1896             }
1897 
1898             tmpImage[i + y * imageWidth] = sum / ((radius + 1) * (radius + 1));
1899         }
1900       }
1901     )
1902 
1903     STRINGIFY(
1904       __kernel void LocalContrastBlurApplyColumn(__global CLPixelType *srcImage, __global CLPixelType *dstImage, __global float *blurImage,
1905           const int radius,
1906           const float strength,
1907           const int imageWidth,
1908           const int imageHeight)
1909       {
1910         const float4 RGB = (float4)(0.2126f, 0.7152f, 0.0722f, 0.0f);
1911 
1912         int x = get_global_id(0);
1913         int y = get_global_id(1);
1914 
1915         if ((x >= imageWidth) || (y >= imageHeight))
1916                 return;
1917 
1918         global float *src = blurImage + x;
1919 
1920         float sum = 0.0f;
1921         float weight = 1.0f;
1922 
1923         int j = y - radius;
1924         while ((j + 7) < y) {
1925             for (int k = 0; k < 8; ++k) // Unroll 8x
1926                 sum += (weight + k) * src[mirrorBottom(j+k) * imageWidth];
1927             weight += 8.0f;
1928             j+=8;
1929         }
1930         while (j < y) {
1931             sum += weight * src[mirrorBottom(j) * imageWidth];
1932             weight += 1.0f;
1933             ++j;
1934         }
1935 
1936         while ((j + 7) < radius + y) {
1937             for (int k = 0; k < 8; ++k) // Unroll 8x
1938                 sum += (weight - k) * src[mirrorTop(j + k, imageHeight) * imageWidth];
1939             weight -= 8.0f;
1940             j+=8;
1941         }
1942         while (j < radius + y) {
1943             sum += weight * src[mirrorTop(j, imageHeight) * imageWidth];
1944             weight -= 1.0f;
1945             ++j;
1946         }
1947 
1948         CLPixelType pixel = srcImage[x + y * imageWidth];
1949         float srcVal = dot(RGB, convert_float4(pixel));
1950         float mult = (srcVal - (sum / ((radius + 1) * (radius + 1)))) * (strength / 100.0f);
1951         mult = (srcVal + mult) / srcVal;
1952 
1953         pixel.x = ClampToQuantum(pixel.x * mult);
1954         pixel.y = ClampToQuantum(pixel.y * mult);
1955         pixel.z = ClampToQuantum(pixel.z * mult);
1956 
1957         dstImage[x + y * imageWidth] = pixel;
1958       }
1959     )
1960 
1961 /*
1962 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1963 %                                                                             %
1964 %                                                                             %
1965 %                                                                             %
1966 %     M o d u l a t e                                                         %
1967 %                                                                             %
1968 %                                                                             %
1969 %                                                                             %
1970 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1971 */
1972 
1973   STRINGIFY(
1974 
1975   static inline void ConvertRGBToHSL(const CLQuantum red,const CLQuantum green, const CLQuantum blue,
1976     float *hue, float *saturation, float *lightness)
1977   {
1978   float
1979     c,
1980     tmax,
1981     tmin;
1982 
1983   /*
1984      Convert RGB to HSL colorspace.
1985      */
1986   tmax=MagickMax(QuantumScale*red,MagickMax(QuantumScale*green, QuantumScale*blue));
1987   tmin=MagickMin(QuantumScale*red,MagickMin(QuantumScale*green, QuantumScale*blue));
1988 
1989   c=tmax-tmin;
1990 
1991   *lightness=(tmax+tmin)/2.0;
1992   if (c <= 0.0)
1993   {
1994     *hue=0.0;
1995     *saturation=0.0;
1996     return;
1997   }
1998 
1999   if (tmax == (QuantumScale*red))
2000   {
2001     *hue=(QuantumScale*green-QuantumScale*blue)/c;
2002     if ((QuantumScale*green) < (QuantumScale*blue))
2003       *hue+=6.0;
2004   }
2005   else
2006     if (tmax == (QuantumScale*green))
2007       *hue=2.0+(QuantumScale*blue-QuantumScale*red)/c;
2008     else
2009       *hue=4.0+(QuantumScale*red-QuantumScale*green)/c;
2010 
2011   *hue*=60.0/360.0;
2012   if (*lightness <= 0.5)
2013     *saturation=c/(2.0*(*lightness));
2014   else
2015     *saturation=c/(2.0-2.0*(*lightness));
2016   }
2017 
2018   static inline void ConvertHSLToRGB(const float hue,const float saturation, const float lightness,
2019       CLQuantum *red,CLQuantum *green,CLQuantum *blue)
2020   {
2021     float
2022       b,
2023       c,
2024       g,
2025       h,
2026       tmin,
2027       r,
2028       x;
2029 
2030     /*
2031        Convert HSL to RGB colorspace.
2032        */
2033     h=hue*360.0;
2034     if (lightness <= 0.5)
2035       c=2.0*lightness*saturation;
2036     else
2037       c=(2.0-2.0*lightness)*saturation;
2038     tmin=lightness-0.5*c;
2039     h-=360.0*floor(h/360.0);
2040     h/=60.0;
2041     x=c*(1.0-fabs(h-2.0*floor(h/2.0)-1.0));
2042     switch ((int) floor(h) % 6)
2043     {
2044       case 0:
2045         {
2046           r=tmin+c;
2047           g=tmin+x;
2048           b=tmin;
2049           break;
2050         }
2051       case 1:
2052         {
2053           r=tmin+x;
2054           g=tmin+c;
2055           b=tmin;
2056           break;
2057         }
2058       case 2:
2059         {
2060           r=tmin;
2061           g=tmin+c;
2062           b=tmin+x;
2063           break;
2064         }
2065       case 3:
2066         {
2067           r=tmin;
2068           g=tmin+x;
2069           b=tmin+c;
2070           break;
2071         }
2072       case 4:
2073         {
2074           r=tmin+x;
2075           g=tmin;
2076           b=tmin+c;
2077           break;
2078         }
2079       case 5:
2080         {
2081           r=tmin+c;
2082           g=tmin;
2083           b=tmin+x;
2084           break;
2085         }
2086       default:
2087         {
2088           r=0.0;
2089           g=0.0;
2090           b=0.0;
2091         }
2092     }
2093     *red=ClampToQuantum(QuantumRange*r);
2094     *green=ClampToQuantum(QuantumRange*g);
2095     *blue=ClampToQuantum(QuantumRange*b);
2096   }
2097 
2098   static inline void ModulateHSL(const float percent_hue, const float percent_saturation,const float percent_lightness,
2099     CLQuantum *red,CLQuantum *green,CLQuantum *blue)
2100   {
2101     float
2102       hue,
2103       lightness,
2104       saturation;
2105 
2106     /*
2107     Increase or decrease color lightness, saturation, or hue.
2108     */
2109     ConvertRGBToHSL(*red,*green,*blue,&hue,&saturation,&lightness);
2110     hue+=0.5*(0.01*percent_hue-1.0);
2111     while (hue < 0.0)
2112       hue+=1.0;
2113     while (hue >= 1.0)
2114       hue-=1.0;
2115     saturation*=0.01*percent_saturation;
2116     lightness*=0.01*percent_lightness;
2117     ConvertHSLToRGB(hue,saturation,lightness,red,green,blue);
2118   }
2119 
2120   __kernel void Modulate(__global CLPixelType *im,
2121     const float percent_brightness,
2122     const float percent_hue,
2123     const float percent_saturation,
2124     const int colorspace)
2125   {
2126 
2127     const int x = get_global_id(0);
2128     const int y = get_global_id(1);
2129     const int columns = get_global_size(0);
2130     const int c = x + y * columns;
2131 
2132     CLPixelType pixel = im[c];
2133 
2134     CLQuantum
2135         blue,
2136         green,
2137         red;
2138 
2139     red=getRed(pixel);
2140     green=getGreen(pixel);
2141     blue=getBlue(pixel);
2142 
2143     switch (colorspace)
2144     {
2145       case HSLColorspace:
2146       default:
2147         {
2148           ModulateHSL(percent_hue, percent_saturation, percent_brightness,
2149               &red, &green, &blue);
2150         }
2151 
2152     }
2153 
2154     CLPixelType filteredPixel;
2155 
2156     setRed(&filteredPixel, red);
2157     setGreen(&filteredPixel, green);
2158     setBlue(&filteredPixel, blue);
2159     filteredPixel.w = pixel.w;
2160 
2161     im[c] = filteredPixel;
2162   }
2163   )
2164 
2165 /*
2166 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2167 %                                                                             %
2168 %                                                                             %
2169 %                                                                             %
2170 %     M o t i o n B l u r                                                     %
2171 %                                                                             %
2172 %                                                                             %
2173 %                                                                             %
2174 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2175 */
2176 
2177   STRINGIFY(
2178     __kernel
2179     void MotionBlur(const __global CLPixelType *input, __global CLPixelType *output,
2180                     const unsigned int imageWidth, const unsigned int imageHeight,
2181                     const __global float *filter, const unsigned int width, const __global int2* offset,
2182                     const float4 bias,
2183                     const ChannelType channel, const unsigned int matte) {
2184 
2185       int2 currentPixel;
2186       currentPixel.x = get_global_id(0);
2187       currentPixel.y = get_global_id(1);
2188 
2189       if (currentPixel.x >= imageWidth
2190           || currentPixel.y >= imageHeight)
2191           return;
2192 
2193       float4 pixel;
2194       pixel.x = (float)bias.x;
2195       pixel.y = (float)bias.y;
2196       pixel.z = (float)bias.z;
2197       pixel.w = (float)bias.w;
2198 
2199       if (((channel & AlphaChannel) == 0) || (matte == 0)) {
2200 
2201         for (int i = 0; i < width; i++) {
2202           // only support EdgeVirtualPixelMethod through ClampToCanvas
2203           // TODO: implement other virtual pixel method
2204           int2 samplePixel = currentPixel + offset[i];
2205           samplePixel.x = ClampToCanvas(samplePixel.x, imageWidth);
2206           samplePixel.y = ClampToCanvas(samplePixel.y, imageHeight);
2207           CLPixelType samplePixelValue = input[ samplePixel.y * imageWidth + samplePixel.x];
2208 
2209           pixel.x += (filter[i] * (float)samplePixelValue.x);
2210           pixel.y += (filter[i] * (float)samplePixelValue.y);
2211           pixel.z += (filter[i] * (float)samplePixelValue.z);
2212           pixel.w += (filter[i] * (float)samplePixelValue.w);
2213         }
2214 
2215         CLPixelType outputPixel;
2216         outputPixel.x = ClampToQuantum(pixel.x);
2217         outputPixel.y = ClampToQuantum(pixel.y);
2218         outputPixel.z = ClampToQuantum(pixel.z);
2219         outputPixel.w = ClampToQuantum(pixel.w);
2220         output[currentPixel.y * imageWidth + currentPixel.x] = outputPixel;
2221       }
2222       else {
2223 
2224         float gamma = 0.0f;
2225         for (int i = 0; i < width; i++) {
2226           // only support EdgeVirtualPixelMethod through ClampToCanvas
2227           // TODO: implement other virtual pixel method
2228           int2 samplePixel = currentPixel + offset[i];
2229           samplePixel.x = ClampToCanvas(samplePixel.x, imageWidth);
2230           samplePixel.y = ClampToCanvas(samplePixel.y, imageHeight);
2231 
2232           CLPixelType samplePixelValue = input[ samplePixel.y * imageWidth + samplePixel.x];
2233 
2234           float alpha = QuantumScale*samplePixelValue.w;
2235           float k = filter[i];
2236           pixel.x = pixel.x + k * alpha * samplePixelValue.x;
2237           pixel.y = pixel.y + k * alpha * samplePixelValue.y;
2238           pixel.z = pixel.z + k * alpha * samplePixelValue.z;
2239 
2240           pixel.w += k * alpha * samplePixelValue.w;
2241 
2242           gamma+=k*alpha;
2243         }
2244         gamma = PerceptibleReciprocal(gamma);
2245         pixel.xyz = gamma*pixel.xyz;
2246 
2247         CLPixelType outputPixel;
2248         outputPixel.x = ClampToQuantum(pixel.x);
2249         outputPixel.y = ClampToQuantum(pixel.y);
2250         outputPixel.z = ClampToQuantum(pixel.z);
2251         outputPixel.w = ClampToQuantum(pixel.w);
2252         output[currentPixel.y * imageWidth + currentPixel.x] = outputPixel;
2253       }
2254     }
2255   )
2256 
2257 /*
2258 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2259 %                                                                             %
2260 %                                                                             %
2261 %                                                                             %
2262 %     R e s i z e                                                             %
2263 %                                                                             %
2264 %                                                                             %
2265 %                                                                             %
2266 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2267 */
2268 
2269   STRINGIFY(
2270   // Based on Box from resize.c
2271   float BoxResizeFilter(const float x)
2272   {
2273     return 1.0f;
2274   }
2275   )
2276 
2277   STRINGIFY(
2278   // Based on CubicBC from resize.c
2279   float CubicBC(const float x,const __global float* resizeFilterCoefficients)
2280   {
2281     /*
2282     Cubic Filters using B,C determined values:
2283     Mitchell-Netravali  B = 1/3 C = 1/3  "Balanced" cubic spline filter
2284     Catmull-Rom         B = 0   C = 1/2  Interpolatory and exact on linears
2285     Spline              B = 1   C = 0    B-Spline Gaussian approximation
2286     Hermite             B = 0   C = 0    B-Spline interpolator
2287 
2288     See paper by Mitchell and Netravali, Reconstruction Filters in Computer
2289     Graphics Computer Graphics, Volume 22, Number 4, August 1988
2290     http://www.cs.utexas.edu/users/fussell/courses/cs384g/lectures/mitchell/
2291     Mitchell.pdf.
2292 
2293     Coefficents are determined from B,C values:
2294     P0 = (  6 - 2*B       )/6 = coeff[0]
2295     P1 =         0
2296     P2 = (-18 +12*B + 6*C )/6 = coeff[1]
2297     P3 = ( 12 - 9*B - 6*C )/6 = coeff[2]
2298     Q0 = (      8*B +24*C )/6 = coeff[3]
2299     Q1 = (    -12*B -48*C )/6 = coeff[4]
2300     Q2 = (      6*B +30*C )/6 = coeff[5]
2301     Q3 = (    - 1*B - 6*C )/6 = coeff[6]
2302 
2303     which are used to define the filter:
2304 
2305     P0 + P1*x + P2*x^2 + P3*x^3      0 <= x < 1
2306     Q0 + Q1*x + Q2*x^2 + Q3*x^3      1 <= x < 2
2307 
2308     which ensures function is continuous in value and derivative (slope).
2309     */
2310     if (x < 1.0)
2311       return(resizeFilterCoefficients[0]+x*(x*
2312       (resizeFilterCoefficients[1]+x*resizeFilterCoefficients[2])));
2313     if (x < 2.0)
2314       return(resizeFilterCoefficients[3]+x*(resizeFilterCoefficients[4]+x*
2315       (resizeFilterCoefficients[5]+x*resizeFilterCoefficients[6])));
2316     return(0.0);
2317   }
2318   )
2319 
2320   STRINGIFY(
2321   float Sinc(const float x)
2322   {
2323     if (x != 0.0f)
2324     {
2325       const float alpha=(float) (MagickPI*x);
2326       return sinpi(x)/alpha;
2327     }
2328     return(1.0f);
2329   }
2330   )
2331 
2332   STRINGIFY(
2333   float Triangle(const float x)
2334   {
2335     /*
2336     1st order (linear) B-Spline, bilinear interpolation, Tent 1D filter, or
2337     a Bartlett 2D Cone filter.  Also used as a Bartlett Windowing function
2338     for Sinc().
2339     */
2340     return ((x<1.0f)?(1.0f-x):0.0f);
2341   }
2342   )
2343 
2344 
2345   STRINGIFY(
2346   float Hann(const float x)
2347   {
2348     /*
2349     Cosine window function:
2350       0.5+0.5*cos(pi*x).
2351     */
2352     const float cosine=cos((MagickPI*x));
2353     return(0.5f+0.5f*cosine);
2354   }
2355   )
2356 
2357   STRINGIFY(
2358   float Hamming(const float x)
2359   {
2360     /*
2361       Offset cosine window function:
2362        .54 + .46 cos(pi x).
2363     */
2364     const float cosine=cos((MagickPI*x));
2365     return(0.54f+0.46f*cosine);
2366   }
2367   )
2368 
2369   STRINGIFY(
2370   float Blackman(const float x)
2371   {
2372     /*
2373       Blackman: 2nd order cosine windowing function:
2374         0.42 + 0.5 cos(pi x) + 0.08 cos(2pi x)
2375 
2376       Refactored by Chantal Racette and Nicolas Robidoux to one trig call and
2377       five flops.
2378     */
2379     const float cosine=cos((MagickPI*x));
2380     return(0.34f+cosine*(0.5f+cosine*0.16f));
2381   }
2382   )
2383 
2384   STRINGIFY(
2385   static inline float applyResizeFilter(const float x, const ResizeWeightingFunctionType filterType, const __global float* filterCoefficients)
2386   {
2387     switch (filterType)
2388     {
2389     /* Call Sinc even for SincFast to get better precision on GPU
2390        and to avoid thread divergence.  Sinc is pretty fast on GPU anyway...*/
2391     case SincWeightingFunction:
2392     case SincFastWeightingFunction:
2393       return Sinc(x);
2394     case CubicBCWeightingFunction:
2395       return CubicBC(x,filterCoefficients);
2396     case BoxWeightingFunction:
2397       return BoxResizeFilter(x);
2398     case TriangleWeightingFunction:
2399       return Triangle(x);
2400     case HannWeightingFunction:
2401       return Hann(x);
2402     case HammingWeightingFunction:
2403       return Hamming(x);
2404     case BlackmanWeightingFunction:
2405       return Blackman(x);
2406 
2407     default:
2408       return 0.0f;
2409     }
2410   }
2411   )
2412 
2413 
2414   STRINGIFY(
2415   static inline float getResizeFilterWeight(const __global float* resizeFilterCubicCoefficients, const ResizeWeightingFunctionType resizeFilterType
2416            , const ResizeWeightingFunctionType resizeWindowType
2417            , const float resizeFilterScale, const float resizeWindowSupport, const float resizeFilterBlur, const float x)
2418   {
2419     float scale;
2420     float xBlur = fabs(x/resizeFilterBlur);
2421     if (resizeWindowSupport < MagickEpsilon
2422         || resizeWindowType == BoxWeightingFunction)
2423     {
2424       scale = 1.0f;
2425     }
2426     else
2427     {
2428       scale = resizeFilterScale;
2429       scale = applyResizeFilter(xBlur*scale, resizeWindowType, resizeFilterCubicCoefficients);
2430     }
2431     float weight = scale * applyResizeFilter(xBlur, resizeFilterType, resizeFilterCubicCoefficients);
2432     return weight;
2433   }
2434 
2435   )
2436 
2437   ;
2438   const char *accelerateKernels2 =
2439 
2440   STRINGIFY(
2441 
2442   static inline unsigned int getNumWorkItemsPerPixel(const unsigned int pixelPerWorkgroup, const unsigned int numWorkItems) {
2443     return (numWorkItems/pixelPerWorkgroup);
2444   }
2445 
2446   // returns the index of the pixel for the current workitem to compute.
2447   // returns -1 if this workitem doesn't need to participate in any computation
2448   static inline int pixelToCompute(const unsigned itemID, const unsigned int pixelPerWorkgroup, const unsigned int numWorkItems) {
2449     const unsigned int numWorkItemsPerPixel = getNumWorkItemsPerPixel(pixelPerWorkgroup, numWorkItems);
2450     int pixelIndex = itemID/numWorkItemsPerPixel;
2451     pixelIndex = (pixelIndex<pixelPerWorkgroup)?pixelIndex:-1;
2452     return pixelIndex;
2453   }
2454 
2455   )
2456 
2457   STRINGIFY(
2458   __kernel __attribute__((reqd_work_group_size(256, 1, 1)))
2459     void ResizeHorizontalFilter(const __global CLQuantum *inputImage, const unsigned int number_channels,
2460       const unsigned int inputColumns, const unsigned int inputRows, __global CLQuantum *filteredImage,
2461       const unsigned int filteredColumns, const unsigned int filteredRows, const float xFactor,
2462       const int resizeFilterType, const int resizeWindowType, const __global float *resizeFilterCubicCoefficients,
2463       const float resizeFilterScale, const float resizeFilterSupport, const float resizeFilterWindowSupport,
2464       const float resizeFilterBlur, __local CLQuantum *inputImageCache, const int numCachedPixels,
2465       const unsigned int pixelPerWorkgroup, const unsigned int pixelChunkSize,
2466       __local float4 *outputPixelCache, __local float *densityCache, __local float *gammaCache)
2467   {
2468     // calculate the range of resized image pixels computed by this workgroup
2469     const unsigned int startX = get_group_id(0)*pixelPerWorkgroup;
2470     const unsigned int stopX = MagickMin(startX + pixelPerWorkgroup,filteredColumns);
2471     const unsigned int actualNumPixelToCompute = stopX - startX;
2472 
2473     // calculate the range of input image pixels to cache
2474     float scale = MagickMax(1.0f/xFactor+MagickEpsilon ,1.0f);
2475     const float support = MagickMax(scale*resizeFilterSupport,0.5f);
2476     scale = PerceptibleReciprocal(scale);
2477 
2478     const int cacheRangeStartX = MagickMax((int)((startX+0.5f)/xFactor+MagickEpsilon-support+0.5f),(int)(0));
2479     const int cacheRangeEndX = MagickMin((int)(cacheRangeStartX + numCachedPixels), (int)inputColumns);
2480 
2481     // cache the input pixels into local memory
2482     const unsigned int y = get_global_id(1);
2483     const unsigned int pos = getPixelIndex(number_channels, inputColumns, cacheRangeStartX, y);
2484     const unsigned int num_elements = (cacheRangeEndX - cacheRangeStartX) * number_channels;
2485     event_t e = async_work_group_copy(inputImageCache, inputImage + pos, num_elements, 0);
2486     wait_group_events(1, &e);
2487 
2488     unsigned int alpha_index = (number_channels == 4) || (number_channels == 2) ? number_channels - 1 : 0;
2489     unsigned int totalNumChunks = (actualNumPixelToCompute+pixelChunkSize-1)/pixelChunkSize;
2490     for (unsigned int chunk = 0; chunk < totalNumChunks; chunk++)
2491     {
2492       const unsigned int chunkStartX = startX + chunk*pixelChunkSize;
2493       const unsigned int chunkStopX = MagickMin(chunkStartX + pixelChunkSize, stopX);
2494       const unsigned int actualNumPixelInThisChunk = chunkStopX - chunkStartX;
2495 
2496       // determine which resized pixel computed by this workitem
2497       const unsigned int itemID = get_local_id(0);
2498       const unsigned int numItems = getNumWorkItemsPerPixel(actualNumPixelInThisChunk, get_local_size(0));
2499 
2500       const int pixelIndex = pixelToCompute(itemID, actualNumPixelInThisChunk, get_local_size(0));
2501 
2502       float4 filteredPixel = (float4)0.0f;
2503       float density = 0.0f;
2504       float gamma = 0.0f;
2505       // -1 means this workitem doesn't participate in the computation
2506       if (pixelIndex != -1)
2507       {
2508         // x coordinated of the resized pixel computed by this workitem
2509         const int x = chunkStartX + pixelIndex;
2510 
2511         // calculate how many steps required for this pixel
2512         const float bisect = (x+0.5)/xFactor+MagickEpsilon;
2513         const unsigned int start = (unsigned int)MagickMax(bisect-support+0.5f,0.0f);
2514         const unsigned int stop  = (unsigned int)MagickMin(bisect+support+0.5f,(float)inputColumns);
2515         const unsigned int n = stop - start;
2516 
2517         // calculate how many steps this workitem will contribute
2518         unsigned int numStepsPerWorkItem = n / numItems;
2519         numStepsPerWorkItem += ((numItems*numStepsPerWorkItem)==n?0:1);
2520 
2521         const unsigned int startStep = (itemID%numItems)*numStepsPerWorkItem;
2522         if (startStep < n)
2523         {
2524           const unsigned int stopStep = MagickMin(startStep+numStepsPerWorkItem, n);
2525 
2526           unsigned int cacheIndex = start+startStep-cacheRangeStartX;
2527           for (unsigned int i = startStep; i < stopStep; i++, cacheIndex++)
2528           {
2529             float weight = getResizeFilterWeight(resizeFilterCubicCoefficients,
2530               (ResizeWeightingFunctionType) resizeFilterType,
2531               (ResizeWeightingFunctionType) resizeWindowType,
2532               resizeFilterScale, resizeFilterWindowSupport,
2533               resizeFilterBlur, scale*(start + i - bisect + 0.5));
2534 
2535             float4 cp = (float4)0.0f;
2536 
2537             __local CLQuantum *p = inputImageCache + (cacheIndex*number_channels);
2538             cp.x = (float) *(p);
2539             if (number_channels > 2)
2540             {
2541               cp.y = (float) *(p + 1);
2542               cp.z = (float) *(p + 2);
2543             }
2544 
2545             if (alpha_index != 0)
2546             {
2547               cp.w = (float) *(p + alpha_index);
2548 
2549               float alpha = weight * QuantumScale * cp.w;
2550 
2551               filteredPixel.x += alpha * cp.x;
2552               filteredPixel.y += alpha * cp.y;
2553               filteredPixel.z += alpha * cp.z;
2554               filteredPixel.w += weight * cp.w;
2555               gamma += alpha;
2556             }
2557             else
2558               filteredPixel += ((float4) weight)*cp;
2559 
2560             density += weight;
2561           }
2562         }
2563       }
2564 
2565       // initialize the accumulators to zero
2566       if (itemID < actualNumPixelInThisChunk) {
2567         outputPixelCache[itemID] = (float4)0.0f;
2568         densityCache[itemID] = 0.0f;
2569         if (alpha_index != 0)
2570           gammaCache[itemID] = 0.0f;
2571       }
2572       barrier(CLK_LOCAL_MEM_FENCE);
2573 
2574       // accumulatte the filtered pixel value and the density
2575       for (unsigned int i = 0; i < numItems; i++) {
2576         if (pixelIndex != -1) {
2577           if (itemID%numItems == i) {
2578             outputPixelCache[pixelIndex]+=filteredPixel;
2579             densityCache[pixelIndex]+=density;
2580             if (alpha_index != 0)
2581               gammaCache[pixelIndex]+=gamma;
2582           }
2583         }
2584         barrier(CLK_LOCAL_MEM_FENCE);
2585       }
2586 
2587       if (itemID < actualNumPixelInThisChunk)
2588       {
2589         float4 filteredPixel = outputPixelCache[itemID];
2590 
2591         float gamma = 0.0f;
2592         if (alpha_index != 0)
2593           gamma = gammaCache[itemID];
2594 
2595         float density = densityCache[itemID];
2596         if ((density != 0.0f) && (density != 1.0f))
2597         {
2598           density = PerceptibleReciprocal(density);
2599           filteredPixel *= (float4) density;
2600           if (alpha_index != 0)
2601             gamma *= density;
2602         }
2603 
2604         if (alpha_index != 0)
2605         {
2606           gamma = PerceptibleReciprocal(gamma);
2607           filteredPixel.x *= gamma;
2608           filteredPixel.y *= gamma;
2609           filteredPixel.z *= gamma;
2610         }
2611 
2612         WriteAllChannels(filteredImage, number_channels, filteredColumns, chunkStartX + itemID, y, filteredPixel);
2613       }
2614     }
2615   }
2616   )
2617 
2618 
2619   STRINGIFY(
2620  __kernel __attribute__((reqd_work_group_size(1, 256, 1)))
2621     void ResizeVerticalFilter(const __global CLQuantum *inputImage, const unsigned int number_channels,
2622       const unsigned int inputColumns, const unsigned int inputRows, __global CLQuantum *filteredImage,
2623       const unsigned int filteredColumns, const unsigned int filteredRows, const float yFactor,
2624       const int resizeFilterType, const int resizeWindowType, const __global float *resizeFilterCubicCoefficients,
2625       const float resizeFilterScale, const float resizeFilterSupport, const float resizeFilterWindowSupport,
2626       const float resizeFilterBlur, __local CLQuantum *inputImageCache, const int numCachedPixels,
2627       const unsigned int pixelPerWorkgroup, const unsigned int pixelChunkSize,
2628       __local float4 *outputPixelCache, __local float *densityCache, __local float *gammaCache)
2629   {
2630     // calculate the range of resized image pixels computed by this workgroup
2631     const unsigned int startY = get_group_id(1)*pixelPerWorkgroup;
2632     const unsigned int stopY = MagickMin(startY + pixelPerWorkgroup,filteredRows);
2633     const unsigned int actualNumPixelToCompute = stopY - startY;
2634 
2635     // calculate the range of input image pixels to cache
2636     float scale = MagickMax(1.0f/yFactor+MagickEpsilon ,1.0f);
2637     const float support = MagickMax(scale*resizeFilterSupport,0.5f);
2638     scale = PerceptibleReciprocal(scale);
2639 
2640     const int cacheRangeStartY = MagickMax((int)((startY+0.5f)/yFactor+MagickEpsilon-support+0.5f),(int)(0));
2641     const int cacheRangeEndY = MagickMin((int)(cacheRangeStartY + numCachedPixels), (int)inputRows);
2642 
2643     // cache the input pixels into local memory
2644     const unsigned int x = get_global_id(0);
2645     unsigned int pos = getPixelIndex(number_channels, inputColumns, x, cacheRangeStartY);
2646     unsigned int rangeLength = cacheRangeEndY-cacheRangeStartY;
2647     unsigned int stride = inputColumns * number_channels;
2648     for (unsigned int i = 0; i < number_channels; i++)
2649     {
2650       event_t e = async_work_group_strided_copy(inputImageCache + (rangeLength*i), inputImage+pos+i, rangeLength, stride, 0);
2651       wait_group_events(1,&e);
2652     }
2653 
2654     unsigned int alpha_index = (number_channels == 4) || (number_channels == 2) ? number_channels - 1 : 0;
2655     unsigned int totalNumChunks = (actualNumPixelToCompute+pixelChunkSize-1)/pixelChunkSize;
2656     for (unsigned int chunk = 0; chunk < totalNumChunks; chunk++)
2657     {
2658       const unsigned int chunkStartY = startY + chunk*pixelChunkSize;
2659       const unsigned int chunkStopY = MagickMin(chunkStartY + pixelChunkSize, stopY);
2660       const unsigned int actualNumPixelInThisChunk = chunkStopY - chunkStartY;
2661 
2662       // determine which resized pixel computed by this workitem
2663       const unsigned int itemID = get_local_id(1);
2664       const unsigned int numItems = getNumWorkItemsPerPixel(actualNumPixelInThisChunk, get_local_size(1));
2665 
2666       const int pixelIndex = pixelToCompute(itemID, actualNumPixelInThisChunk, get_local_size(1));
2667 
2668       float4 filteredPixel = (float4)0.0f;
2669       float density = 0.0f;
2670       float gamma = 0.0f;
2671       // -1 means this workitem doesn't participate in the computation
2672       if (pixelIndex != -1)
2673       {
2674         // x coordinated of the resized pixel computed by this workitem
2675         const int y = chunkStartY + pixelIndex;
2676 
2677         // calculate how many steps required for this pixel
2678         const float bisect = (y+0.5)/yFactor+MagickEpsilon;
2679         const unsigned int start = (unsigned int)MagickMax(bisect-support+0.5f,0.0f);
2680         const unsigned int stop  = (unsigned int)MagickMin(bisect+support+0.5f,(float)inputRows);
2681         const unsigned int n = stop - start;
2682 
2683         // calculate how many steps this workitem will contribute
2684         unsigned int numStepsPerWorkItem = n / numItems;
2685         numStepsPerWorkItem += ((numItems*numStepsPerWorkItem)==n?0:1);
2686 
2687         const unsigned int startStep = (itemID%numItems)*numStepsPerWorkItem;
2688         if (startStep < n)
2689         {
2690           const unsigned int stopStep = MagickMin(startStep+numStepsPerWorkItem, n);
2691 
2692           unsigned int cacheIndex = start+startStep-cacheRangeStartY;
2693           for (unsigned int i = startStep; i < stopStep; i++, cacheIndex++)
2694           {
2695             float weight = getResizeFilterWeight(resizeFilterCubicCoefficients,
2696               (ResizeWeightingFunctionType) resizeFilterType,
2697               (ResizeWeightingFunctionType) resizeWindowType,
2698               resizeFilterScale, resizeFilterWindowSupport,
2699               resizeFilterBlur, scale*(start + i - bisect + 0.5));
2700 
2701             float4 cp = (float4)0.0f;
2702 
2703             __local CLQuantum *p = inputImageCache + cacheIndex;
2704             cp.x = (float) *(p);
2705             if (number_channels > 2)
2706             {
2707               cp.y = (float) *(p + rangeLength);
2708               cp.z = (float) *(p + (rangeLength * 2));
2709             }
2710 
2711             if (alpha_index != 0)
2712             {
2713               cp.w = (float) *(p + (rangeLength * alpha_index));
2714 
2715               float alpha = weight * QuantumScale * cp.w;
2716 
2717               filteredPixel.x += alpha * cp.x;
2718               filteredPixel.y += alpha * cp.y;
2719               filteredPixel.z += alpha * cp.z;
2720               filteredPixel.w += weight * cp.w;
2721               gamma += alpha;
2722             }
2723             else
2724               filteredPixel += ((float4) weight)*cp;
2725 
2726             density += weight;
2727           }
2728         }
2729       }
2730 
2731       // initialize the accumulators to zero
2732       if (itemID < actualNumPixelInThisChunk) {
2733         outputPixelCache[itemID] = (float4)0.0f;
2734         densityCache[itemID] = 0.0f;
2735         if (alpha_index != 0)
2736           gammaCache[itemID] = 0.0f;
2737       }
2738       barrier(CLK_LOCAL_MEM_FENCE);
2739 
2740       // accumulatte the filtered pixel value and the density
2741       for (unsigned int i = 0; i < numItems; i++) {
2742         if (pixelIndex != -1) {
2743           if (itemID%numItems == i) {
2744             outputPixelCache[pixelIndex]+=filteredPixel;
2745             densityCache[pixelIndex]+=density;
2746             if (alpha_index != 0)
2747               gammaCache[pixelIndex]+=gamma;
2748           }
2749         }
2750         barrier(CLK_LOCAL_MEM_FENCE);
2751       }
2752 
2753       if (itemID < actualNumPixelInThisChunk)
2754       {
2755         float4 filteredPixel = outputPixelCache[itemID];
2756 
2757         float gamma = 0.0f;
2758         if (alpha_index != 0)
2759           gamma = gammaCache[itemID];
2760 
2761         float density = densityCache[itemID];
2762         if ((density != 0.0f) && (density != 1.0f))
2763         {
2764           density = PerceptibleReciprocal(density);
2765           filteredPixel *= (float4) density;
2766           if (alpha_index != 0)
2767             gamma *= density;
2768         }
2769 
2770         if (alpha_index != 0)
2771         {
2772           gamma = PerceptibleReciprocal(gamma);
2773           filteredPixel.x *= gamma;
2774           filteredPixel.y *= gamma;
2775           filteredPixel.z *= gamma;
2776         }
2777 
2778         WriteAllChannels(filteredImage, number_channels, filteredColumns, x, chunkStartY + itemID, filteredPixel);
2779       }
2780     }
2781   }
2782   )
2783 
2784 /*
2785 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2786 %                                                                             %
2787 %                                                                             %
2788 %                                                                             %
2789 %     R o t a t i o n a l B l u r                                             %
2790 %                                                                             %
2791 %                                                                             %
2792 %                                                                             %
2793 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2794 */
2795 
2796   STRINGIFY(
2797   __kernel void RotationalBlur(const __global CLQuantum *image,
2798     const unsigned int number_channels,const unsigned int channel,
2799     const float2 blurCenter,__constant float *cos_theta,
2800     __constant float *sin_theta,const unsigned int cossin_theta_size,
2801     __global CLQuantum *filteredImage)
2802   {
2803     const int x = get_global_id(0);
2804     const int y = get_global_id(1);
2805     const int columns = get_global_size(0);
2806     const int rows = get_global_size(1);
2807     unsigned int step = 1;
2808     float center_x = (float) x - blurCenter.x;
2809     float center_y = (float) y - blurCenter.y;
2810     float radius = hypot(center_x, center_y);
2811 
2812     float blur_radius = hypot(blurCenter.x, blurCenter.y);
2813 
2814     if (radius > MagickEpsilon)
2815     {
2816       step = (unsigned int) (blur_radius / radius);
2817       if (step == 0)
2818         step = 1;
2819       if (step >= cossin_theta_size)
2820         step = cossin_theta_size-1;
2821     }
2822 
2823     float4 result = 0.0f;
2824     float normalize = 0.0f;
2825     float gamma = 0.0f;
2826 
2827     for (unsigned int i=0; i<cossin_theta_size; i+=step)
2828     {
2829       int cx = ClampToCanvas(blurCenter.x+center_x*cos_theta[i]-center_y*sin_theta[i]+0.5f,columns);
2830       int cy = ClampToCanvas(blurCenter.y+center_x*sin_theta[i]+center_y*cos_theta[i]+0.5f,rows);
2831 
2832       float4 pixel = ReadAllChannels(image, number_channels, columns, cx, cy);
2833 
2834       if ((number_channels == 4) || (number_channels == 2))
2835       {
2836         float alpha = (float)(QuantumScale*pixel.w);
2837 
2838         gamma += alpha;
2839 
2840         result.x += alpha * pixel.x;
2841         result.y += alpha * pixel.y;
2842         result.z += alpha * pixel.z;
2843         result.w += pixel.w;
2844       }
2845       else
2846         result += pixel;
2847 
2848       normalize += 1.0f;
2849     }
2850 
2851     normalize = PerceptibleReciprocal(normalize);
2852 
2853     if ((number_channels == 4) || (number_channels == 2))
2854     {
2855       gamma = PerceptibleReciprocal(gamma);
2856       result.x *= gamma;
2857       result.y *= gamma;
2858       result.z *= gamma;
2859       result.w *= normalize;
2860     }
2861     else
2862       result *= normalize;
2863 
2864     WriteFloat4(filteredImage, number_channels, columns, x, y, channel, result);
2865   }
2866   )
2867 
2868 /*
2869 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2870 %                                                                             %
2871 %                                                                             %
2872 %                                                                             %
2873 %     U n s h a r p M a s k                                                   %
2874 %                                                                             %
2875 %                                                                             %
2876 %                                                                             %
2877 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2878 */
2879 
2880   STRINGIFY(
2881   __kernel void UnsharpMaskBlurColumn(const __global CLQuantum* image,
2882     const __global float4 *blurRowData,const unsigned int number_channels,
2883     const ChannelType channel,const unsigned int columns,
2884     const unsigned int rows,__local float4* cachedData,
2885     __local float* cachedFilter,const __global float *filter,
2886     const unsigned int width,const float gain, const float threshold,
2887     __global CLQuantum *filteredImage)
2888   {
2889     const unsigned int radius = (width-1)/2;
2890 
2891     // cache the pixel shared by the workgroup
2892     const int groupX = get_group_id(0);
2893     const int groupStartY = get_group_id(1)*get_local_size(1) - radius;
2894     const int groupStopY = (get_group_id(1)+1)*get_local_size(1) + radius;
2895 
2896     if ((groupStartY >= 0) && (groupStopY < rows))
2897     {
2898       event_t e = async_work_group_strided_copy(cachedData,
2899         blurRowData+groupStartY*columns+groupX,groupStopY-groupStartY,columns,0);
2900       wait_group_events(1,&e);
2901     }
2902     else
2903     {
2904       for (int i = get_local_id(1); i < (groupStopY - groupStartY); i+=get_local_size(1))
2905         cachedData[i] = blurRowData[ClampToCanvas(groupStartY+i,rows)*columns + groupX];
2906 
2907       barrier(CLK_LOCAL_MEM_FENCE);
2908     }
2909     // cache the filter as well
2910     event_t e = async_work_group_copy(cachedFilter,filter,width,0);
2911     wait_group_events(1,&e);
2912 
2913     // only do the work if this is not a patched item
2914     const int cy = get_global_id(1);
2915 
2916     if (cy < rows)
2917     {
2918       float4 blurredPixel = (float4) 0.0f;
2919 
2920       int i = 0;
2921 
2922       for ( ; i+7 < width; )
2923       {
2924         for (int j=0; j < 8; j++, i++)
2925           blurredPixel+=cachedFilter[i+j]*cachedData[i+j+get_local_id(1)];
2926       }
2927 
2928       for ( ; i < width; i++)
2929         blurredPixel+=cachedFilter[i]*cachedData[i+get_local_id(1)];
2930 
2931       float4 inputImagePixel = ReadFloat4(image,number_channels,columns,groupX,cy,channel);
2932       float4 outputPixel = inputImagePixel - blurredPixel;
2933 
2934       float quantumThreshold = QuantumRange*threshold;
2935 
2936       int4 mask = isless(fabs(2.0f*outputPixel), (float4)quantumThreshold);
2937       outputPixel = select(inputImagePixel + outputPixel * gain, inputImagePixel, mask);
2938 
2939       //write back
2940       WriteFloat4(filteredImage,number_channels,columns,groupX,cy,channel,outputPixel);
2941     }
2942   }
2943   )
2944 
2945   STRINGIFY(
2946   __kernel void UnsharpMask(const __global CLQuantum *image,const unsigned int number_channels,
2947     const ChannelType channel,__constant float *filter,const unsigned int width,
2948     const unsigned int columns,const unsigned int rows,__local float4 *pixels,
2949     const float gain,const float threshold,__global CLQuantum *filteredImage)
2950   {
2951     const unsigned int x = get_global_id(0);
2952     const unsigned int y = get_global_id(1);
2953 
2954     const unsigned int radius = (width - 1) / 2;
2955 
2956     int row = y - radius;
2957     int baseRow = get_group_id(1) * get_local_size(1) - radius;
2958     int endRow = (get_group_id(1) + 1) * get_local_size(1) + radius;
2959 
2960     while (row < endRow) {
2961       int srcy = (row < 0) ? -row : row; // mirror pad
2962       srcy = (srcy >= rows) ? (2 * rows - srcy - 1) : srcy;
2963 
2964       float4 value = 0.0f;
2965 
2966       int ix = x - radius;
2967       int i = 0;
2968 
2969       while (i + 7 < width) {
2970         for (int j = 0; j < 8; ++j) { // unrolled
2971           int srcx = ix + j;
2972           srcx = (srcx < 0) ? -srcx : srcx;
2973           srcx = (srcx >= columns) ? (2 * columns - srcx - 1) : srcx;
2974           value += filter[i + j] * ReadFloat4(image, number_channels, columns, srcx, srcy, channel);
2975         }
2976         ix += 8;
2977         i += 8;
2978       }
2979 
2980       while (i < width) {
2981         int srcx = (ix < 0) ? -ix : ix; // mirror pad
2982         srcx = (srcx >= columns) ? (2 * columns - srcx - 1) : srcx;
2983         value += filter[i] * ReadFloat4(image, number_channels, columns, srcx, srcy, channel);
2984         ++i;
2985         ++ix;
2986       }
2987       pixels[(row - baseRow) * get_local_size(0) + get_local_id(0)] = value;
2988       row += get_local_size(1);
2989     }
2990 
2991     barrier(CLK_LOCAL_MEM_FENCE);
2992 
2993     const int px = get_local_id(0);
2994     const int py = get_local_id(1);
2995     const int prp = get_local_size(0);
2996     float4 value = (float4)(0.0f);
2997 
2998     int i = 0;
2999     while (i + 7 < width) {
3000       for (int j = 0; j < 8; ++j) // unrolled
3001         value += (float4)(filter[i]) * pixels[px + (py + i + j) * prp];
3002       i += 8;
3003     }
3004     while (i < width) {
3005       value += (float4)(filter[i]) * pixels[px + (py + i) * prp];
3006       ++i;
3007     }
3008 
3009     if ((x < columns) && (y < rows)) {
3010       float4 srcPixel = ReadFloat4(image, number_channels, columns, x, y, channel);
3011       float4 diff = srcPixel - value;
3012 
3013       float quantumThreshold = QuantumRange*threshold;
3014 
3015       int4 mask = isless(fabs(2.0f * diff), (float4)quantumThreshold);
3016       value = select(srcPixel + diff * gain, srcPixel, mask);
3017 
3018       WriteFloat4(filteredImage, number_channels, columns, x, y, channel, value);
3019     }
3020   }
3021   )
3022 
3023   STRINGIFY(
3024     __kernel __attribute__((reqd_work_group_size(64, 4, 1)))
3025     void WaveletDenoise(__global CLQuantum *srcImage,__global CLQuantum *dstImage,
3026       const unsigned int number_channels,const unsigned int max_channels,
3027       const float threshold,const int passes,const unsigned int imageWidth,
3028       const unsigned int imageHeight)
3029   {
3030     const int pad = (1 << (passes - 1));
3031     const int tileSize = 64;
3032     const int tileRowPixels = 64;
3033     const float noise[] = { 0.8002, 0.2735, 0.1202, 0.0585, 0.0291, 0.0152, 0.0080, 0.0044 };
3034 
3035     CLQuantum stage[48]; // 16 * 3 (we only need 3 channels)
3036 
3037     local float buffer[64 * 64];
3038 
3039     int srcx = (get_group_id(0) + get_global_offset(0) / tileSize) * (tileSize - 2 * pad) - pad + get_local_id(0);
3040     int srcy = (get_group_id(1) + get_global_offset(1) / 4) * (tileSize - 2 * pad) - pad;
3041 
3042     for (int i = get_local_id(1); i < tileSize; i += get_local_size(1)) {
3043       int pos = (mirrorTop(mirrorBottom(srcx), imageWidth) * number_channels) +
3044                 (mirrorTop(mirrorBottom(srcy + i), imageHeight)) * imageWidth * number_channels;
3045 
3046       for (int channel = 0; channel < max_channels; ++channel)
3047         stage[(i / 4) + (16 * channel)] = srcImage[pos + channel];
3048     }
3049 
3050     for (int channel = 0; channel < max_channels; ++channel) {
3051       // Load LDS
3052       for (int i = get_local_id(1); i < tileSize; i += get_local_size(1))
3053         buffer[get_local_id(0) + i * tileRowPixels] = convert_float(stage[(i / 4) + (16 * channel)]);
3054 
3055       // Process
3056 
3057       float tmp[16];
3058       float accum[16];
3059       float pixel;
3060 
3061       for (int i = 0; i < 16; i++)
3062         accum[i]=0.0f;
3063 
3064       for (int pass = 0; pass < passes; ++pass) {
3065         const int radius = 1 << pass;
3066         const int x = get_local_id(0);
3067         const float thresh = threshold * noise[pass];
3068 
3069         // Apply horizontal hat
3070         for (int i = get_local_id(1); i < tileSize; i += get_local_size(1)) {
3071           const int offset = i * tileRowPixels;
3072           if (pass == 0)
3073             tmp[i / 4] = buffer[x + offset]; // snapshot input on first pass
3074           pixel = 0.5f * tmp[i / 4] + 0.25 * (buffer[mirrorBottom(x - radius) + offset] + buffer[mirrorTop(x + radius, tileSize) + offset]);
3075           barrier(CLK_LOCAL_MEM_FENCE);
3076           buffer[x + offset] = pixel;
3077         }
3078         barrier(CLK_LOCAL_MEM_FENCE);
3079 
3080         // Apply vertical hat
3081         for (int i = get_local_id(1); i < tileSize; i += get_local_size(1)) {
3082           pixel = 0.5f * buffer[x + i * tileRowPixels] + 0.25 * (buffer[x + mirrorBottom(i - radius) * tileRowPixels] + buffer[x + mirrorTop(i + radius, tileRowPixels) * tileRowPixels]);
3083           float delta = tmp[i / 4] - pixel;
3084           tmp[i / 4] = pixel; // hold output in tmp until all workitems are done
3085           if (delta < -thresh)
3086             delta += thresh;
3087           else if (delta > thresh)
3088             delta -= thresh;
3089           else
3090             delta = 0;
3091           accum[i / 4] += delta;
3092         }
3093         barrier(CLK_LOCAL_MEM_FENCE);
3094 
3095         if (pass < passes - 1)
3096           for (int i = get_local_id(1); i < tileSize; i += get_local_size(1))
3097             buffer[x + i * tileRowPixels] = tmp[i / 4]; // store lowpass for next pass
3098         else  // last pass
3099           for (int i = get_local_id(1); i < tileSize; i += get_local_size(1))
3100             accum[i / 4] += tmp[i / 4]; // add the lowpass signal back to output
3101         barrier(CLK_LOCAL_MEM_FENCE);
3102       }
3103 
3104       for (int i = get_local_id(1); i < tileSize; i += get_local_size(1))
3105         stage[(i / 4) + (16 * channel)] = ClampToQuantum(accum[i / 4]);
3106 
3107       barrier(CLK_LOCAL_MEM_FENCE);
3108     }
3109 
3110     // Write from stage to output
3111 
3112     if ((get_local_id(0) >= pad) && (get_local_id(0) < tileSize - pad) && (srcx >= 0) && (srcx < imageWidth)) {
3113       for (int i = get_local_id(1); i < tileSize; i += get_local_size(1)) {
3114         if ((i >= pad) && (i < tileSize - pad) && (srcy + i >= 0) && (srcy + i < imageHeight)) {
3115           int pos = (srcx * number_channels) + ((srcy + i) * (imageWidth * number_channels));
3116           for (int channel = 0; channel < max_channels; ++channel) {
3117             dstImage[pos + channel] = stage[(i / 4) + (16 * channel)];
3118           }
3119         }
3120       }
3121     }
3122   }
3123   )
3124 
3125   ;
3126 
3127 #endif // MAGICKCORE_OPENCL_SUPPORT
3128 
3129 #if defined(__cplusplus) || defined(c_plusplus)
3130 }
3131 #endif
3132 
3133 #endif // MAGICKCORE_ACCELERATE_KERNELS_PRIVATE_H
3134