blob: 1d23723a8fb005024d7f6d34d8c575dc81b9dfcd [file] [log] [blame]
cristy3ed852e2009-09-05 21:47:34 +00001/*
2%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
3% %
4% %
5% %
6% %
7% V V AAA L IIIII DDDD AAA TTTTT EEEEE %
8% V V A A L I D D A A T E %
9% V V AAAAA L I D D AAAAA T EEE %
10% V V A A L I D D A A T E %
11% V A A LLLLL IIIII DDDD A A T EEEEE %
12% %
13% %
14% ImageMagick Validation Suite %
15% %
16% Software Design %
cristyde984cd2013-12-01 14:49:27 +000017% Cristy %
cristy3ed852e2009-09-05 21:47:34 +000018% March 2001 %
19% %
20% %
Cristyf6ff9ea2016-12-05 09:53:35 -050021% Copyright 1999-2017 ImageMagick Studio LLC, a non-profit organization %
cristy3ed852e2009-09-05 21:47:34 +000022% dedicated to making software imaging solutions freely available. %
23% %
24% You may not use this file except in compliance with the License. You may %
25% obtain a copy of the License at %
26% %
Cristyf19d4142017-04-24 11:34:30 -040027% https://www.imagemagick.org/script/license.php %
cristy3ed852e2009-09-05 21:47:34 +000028% %
29% Unless required by applicable law or agreed to in writing, software %
30% distributed under the License is distributed on an "AS IS" BASIS, %
31% WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. %
32% see the License for the specific language governing permissions and %
33% limitations under the License. %
34% %
35%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
36%
37%
38*/
39
40/*
41 Include declarations.
42*/
43#include <stdio.h>
cristy5e8e9812009-12-17 15:13:38 +000044#include <stdlib.h>
cristy3ed852e2009-09-05 21:47:34 +000045#include <string.h>
cristy5e8e9812009-12-17 15:13:38 +000046#include <ctype.h>
cristy3ed852e2009-09-05 21:47:34 +000047#include <math.h>
cristye2605742012-12-12 11:18:34 +000048#include <locale.h>
cristy9883b452011-07-01 23:37:21 +000049#include "MagickWand/MagickWand.h"
cristy510d06a2011-07-06 23:43:54 +000050#include "MagickCore/colorspace-private.h"
dirk775194c2014-05-17 19:47:54 +000051#include "MagickCore/gem.h"
glennrp4084cda2012-12-11 14:13:35 +000052#include "MagickCore/resource_.h"
cristy9883b452011-07-01 23:37:21 +000053#include "MagickCore/string-private.h"
cristy3ed852e2009-09-05 21:47:34 +000054#include "validate.h"
55
56/*
cristy8d24e062013-05-07 12:48:19 +000057 Define declarations.
58*/
59#define CIEEpsilon (216.0/24389.0)
60#define CIEK (24389.0/27.0)
61#define D65X 0.950456
62#define D65Y 1.0
63#define D65Z 1.088754
cristya7230ee2013-07-25 11:51:29 +000064#define ReferenceEpsilon (QuantumRange*1.0e-2)
cristy8d24e062013-05-07 12:48:19 +000065
66/*
cristy3ed852e2009-09-05 21:47:34 +000067%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
68% %
69% %
70% %
cristy572e0972013-05-07 12:51:28 +000071% V a l i d a t e C o l o r s p a c e s %
cristy80e870e2013-05-07 12:23:02 +000072% %
73% %
74% %
75%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
76%
cristy572e0972013-05-07 12:51:28 +000077% ValidateColorspaces() validates the ImageMagick colorspaces and returns the
78% number of validation tests that passed and failed.
cristy80e870e2013-05-07 12:23:02 +000079%
cristy572e0972013-05-07 12:51:28 +000080% The format of the ValidateColorspaces method is:
cristy80e870e2013-05-07 12:23:02 +000081%
cristy572e0972013-05-07 12:51:28 +000082% size_t ValidateColorspaces(ImageInfo *image_info,size_t *fail,
83% ExceptionInfo *exception)
cristy80e870e2013-05-07 12:23:02 +000084%
85% A description of each parameter follows:
86%
87% o image_info: the image info.
88%
cristy80e870e2013-05-07 12:23:02 +000089% o fail: return the number of validation tests that pass.
90%
91% o exception: return any errors or warnings in this structure.
92%
93*/
cristy8d24e062013-05-07 12:48:19 +000094
95static void ConvertHSIToRGB(const double hue,const double saturation,
96 const double intensity,double *red,double *green,double *blue)
97{
98 double
99 h;
100
cristy8d24e062013-05-07 12:48:19 +0000101 h=360.0*hue;
102 h-=360.0*floor(h/360.0);
103 if (h < 120.0)
104 {
105 *blue=intensity*(1.0-saturation);
106 *red=intensity*(1.0+saturation*cos(h*(MagickPI/180.0))/cos((60.0-h)*
107 (MagickPI/180.0)));
108 *green=3.0*intensity-*red-*blue;
109 }
110 else
111 if (h < 240.0)
112 {
113 h-=120.0;
114 *red=intensity*(1.0-saturation);
115 *green=intensity*(1.0+saturation*cos(h*(MagickPI/180.0))/cos((60.0-h)*
116 (MagickPI/180.0)));
117 *blue=3.0*intensity-*red-*green;
118 }
119 else
120 {
121 h-=240.0;
122 *green=intensity*(1.0-saturation);
123 *blue=intensity*(1.0+saturation*cos(h*(MagickPI/180.0))/cos((60.0-h)*
124 (MagickPI/180.0)));
125 *red=3.0*intensity-*green-*blue;
126 }
127 *red*=QuantumRange;
128 *green*=QuantumRange;
129 *blue*=QuantumRange;
130}
131
cristy8d24e062013-05-07 12:48:19 +0000132static void ConvertRGBToHSI(const double red,const double green,
133 const double blue,double *hue,double *saturation,double *intensity)
134{
135 double
136 alpha,
137 beta;
138
cristy8d24e062013-05-07 12:48:19 +0000139 *intensity=(QuantumScale*red+QuantumScale*green+QuantumScale*blue)/3.0;
140 if (*intensity <= 0.0)
141 {
142 *hue=0.0;
143 *saturation=0.0;
144 return;
145 }
146 *saturation=1.0-MagickMin(QuantumScale*red,MagickMin(QuantumScale*green,
147 QuantumScale*blue))/(*intensity);
148 alpha=0.5*(2.0*QuantumScale*red-QuantumScale*green-QuantumScale*blue);
149 beta=0.8660254037844385*(QuantumScale*green-QuantumScale*blue);
150 *hue=atan2(beta,alpha)*(180.0/MagickPI)/360.0;
151 if (*hue < 0.0)
152 *hue+=1.0;
153}
154
cristy8d24e062013-05-07 12:48:19 +0000155static void ConvertHSVToRGB(const double hue,const double saturation,
156 const double value,double *red,double *green,double *blue)
157{
158 double
159 c,
160 h,
161 min,
162 x;
163
cristy8d24e062013-05-07 12:48:19 +0000164 h=hue*360.0;
165 c=value*saturation;
166 min=value-c;
167 h-=360.0*floor(h/360.0);
168 h/=60.0;
169 x=c*(1.0-fabs(h-2.0*floor(h/2.0)-1.0));
170 switch ((int) floor(h))
171 {
172 case 0:
173 {
174 *red=QuantumRange*(min+c);
175 *green=QuantumRange*(min+x);
176 *blue=QuantumRange*min;
177 break;
178 }
179 case 1:
180 {
181 *red=QuantumRange*(min+x);
182 *green=QuantumRange*(min+c);
183 *blue=QuantumRange*min;
184 break;
185 }
186 case 2:
187 {
188 *red=QuantumRange*min;
189 *green=QuantumRange*(min+c);
190 *blue=QuantumRange*(min+x);
191 break;
192 }
193 case 3:
194 {
195 *red=QuantumRange*min;
196 *green=QuantumRange*(min+x);
197 *blue=QuantumRange*(min+c);
198 break;
199 }
200 case 4:
201 {
202 *red=QuantumRange*(min+x);
203 *green=QuantumRange*min;
204 *blue=QuantumRange*(min+c);
205 break;
206 }
207 case 5:
208 {
209 *red=QuantumRange*(min+c);
210 *green=QuantumRange*min;
211 *blue=QuantumRange*(min+x);
212 break;
213 }
214 default:
215 {
216 *red=0.0;
217 *green=0.0;
218 *blue=0.0;
219 }
220 }
221}
222
223static inline void ConvertRGBToXYZ(const double red,const double green,
224 const double blue,double *X,double *Y,double *Z)
225{
226 double
227 b,
228 g,
229 r;
230
cristy8d24e062013-05-07 12:48:19 +0000231 r=QuantumScale*DecodePixelGamma(red);
232 g=QuantumScale*DecodePixelGamma(green);
233 b=QuantumScale*DecodePixelGamma(blue);
234 *X=0.41239558896741421610*r+0.35758343076371481710*g+0.18049264738170157350*b;
235 *Y=0.21258623078559555160*r+0.71517030370341084990*g+0.07220049864333622685*b;
236 *Z=0.01929721549174694484*r+0.11918386458084853180*g+0.95049712513157976600*b;
237}
238
239static inline void ConvertXYZToLab(const double X,const double Y,const double Z,
240 double *L,double *a,double *b)
241{
242 double
243 x,
244 y,
245 z;
246
247 if ((X/D65X) > CIEEpsilon)
248 x=pow(X/D65X,1.0/3.0);
249 else
250 x=(CIEK*X/D65X+16.0)/116.0;
251 if ((Y/D65Y) > CIEEpsilon)
252 y=pow(Y/D65Y,1.0/3.0);
253 else
254 y=(CIEK*Y/D65Y+16.0)/116.0;
255 if ((Z/D65Z) > CIEEpsilon)
256 z=pow(Z/D65Z,1.0/3.0);
257 else
258 z=(CIEK*Z/D65Z+16.0)/116.0;
259 *L=((116.0*y)-16.0)/100.0;
260 *a=(500.0*(x-y))/255.0+0.5;
261 *b=(200.0*(y-z))/255.0+0.5;
262}
263
264static void ConvertRGBToLab(const double red,const double green,
265 const double blue,double *L,double *a,double *b)
266{
267 double
268 X,
269 Y,
270 Z;
271
272 ConvertRGBToXYZ(red,green,blue,&X,&Y,&Z);
273 ConvertXYZToLab(X,Y,Z,L,a,b);
274}
275
276static inline void ConvertLabToXYZ(const double L,const double a,const double b,
277 double *X,double *Y,double *Z)
278{
279 double
280 x,
281 y,
282 z;
283
284 y=(L+16.0)/116.0;
285 x=y+a/500.0;
286 z=y-b/200.0;
287 if ((x*x*x) > CIEEpsilon)
288 x=(x*x*x);
289 else
290 x=(116.0*x-16.0)/CIEK;
291 if ((y*y*y) > CIEEpsilon)
292 y=(y*y*y);
293 else
294 y=L/CIEK;
295 if ((z*z*z) > CIEEpsilon)
296 z=(z*z*z);
297 else
298 z=(116.0*z-16.0)/CIEK;
299 *X=D65X*x;
300 *Y=D65Y*y;
301 *Z=D65Z*z;
302}
303
304static inline void ConvertXYZToRGB(const double x,const double y,const double z,
305 double *red,double *green,double *blue)
306{
307 double
308 b,
309 g,
310 r;
311
cristy8d24e062013-05-07 12:48:19 +0000312 r=3.2406*x-1.5372*y-0.4986*z;
313 g=(-0.9689*x+1.8758*y+0.0415*z);
314 b=0.0557*x-0.2040*y+1.0570*z;
315 *red=EncodePixelGamma(QuantumRange*r);
316 *green=EncodePixelGamma(QuantumRange*g);
317 *blue=EncodePixelGamma(QuantumRange*b);
318}
319
320static inline void ConvertLabToRGB(const double L,const double a,
321 const double b,double *red,double *green,double *blue)
322{
323 double
324 X,
325 Y,
326 Z;
327
328 ConvertLabToXYZ(L*100.0,255.0*(a-0.5),255.0*(b-0.5),&X,&Y,&Z);
329 ConvertXYZToRGB(X,Y,Z,red,green,blue);
330}
331
332static void ConvertRGBToYPbPr(const double red,const double green,
333 const double blue,double *Y,double *Pb,double *Pr)
334{
cristy8d24e062013-05-07 12:48:19 +0000335 *Y=QuantumScale*(0.298839*red+0.586811*green+0.114350*blue);
336 *Pb=QuantumScale*((-0.1687367)*red-0.331264*green+0.5*blue)+0.5;
337 *Pr=QuantumScale*(0.5*red-0.418688*green-0.081312*blue)+0.5;
338}
339
340static void ConvertRGBToYCbCr(const double red,const double green,
341 const double blue,double *Y,double *Cb,double *Cr)
342{
cristy8d24e062013-05-07 12:48:19 +0000343 ConvertRGBToYPbPr(red,green,blue,Y,Cb,Cr);
344}
345
346static void ConvertYPbPrToRGB(const double Y,const double Pb,const double Pr,
347 double *red,double *green,double *blue)
348{
cristy8d24e062013-05-07 12:48:19 +0000349 *red=QuantumRange*(0.99999999999914679361*Y-1.2188941887145875e-06*(Pb-0.5)+
350 1.4019995886561440468*(Pr-0.5));
351 *green=QuantumRange*(0.99999975910502514331*Y-0.34413567816504303521*(Pb-0.5)-
352 0.71413649331646789076*(Pr-0.5));
353 *blue=QuantumRange*(1.00000124040004623180*Y+1.77200006607230409200*(Pb-0.5)+
354 2.1453384174593273e-06*(Pr-0.5));
355}
356
357static void ConvertYCbCrToRGB(const double Y,const double Cb,
358 const double Cr,double *red,double *green,double *blue)
359{
cristy8d24e062013-05-07 12:48:19 +0000360 ConvertYPbPrToRGB(Y,Cb,Cr,red,green,blue);
361}
362
363static inline void ConvertLCHabToXYZ(const double luma,const double chroma,
364 const double hue,double *X,double *Y,double *Z)
365{
366 ConvertLabToXYZ(luma,chroma*cos(hue*MagickPI/180.0),chroma*
367 sin(hue*MagickPI/180.0),X,Y,Z);
368}
369
370static void ConvertLCHabToRGB(const double luma,const double chroma,
371 const double hue,double *red,double *green,double *blue)
372{
373 double
374 X,
375 Y,
376 Z;
377
cristy6d897c72013-06-19 19:20:40 +0000378 ConvertLCHabToXYZ(luma*100.0,255.0*(chroma-0.5),360.0*hue,&X,&Y,&Z);
cristy8d24e062013-05-07 12:48:19 +0000379 ConvertXYZToRGB(X,Y,Z,red,green,blue);
380}
381
382static void ConvertRGBToHSV(const double red,const double green,
383 const double blue,double *hue,double *saturation,double *value)
384{
385 double
386 c,
387 max,
388 min;
389
cristy8d24e062013-05-07 12:48:19 +0000390 max=MagickMax(QuantumScale*red,MagickMax(QuantumScale*green,
391 QuantumScale*blue));
392 min=MagickMin(QuantumScale*red,MagickMin(QuantumScale*green,
393 QuantumScale*blue));
394 c=max-min;
395 *value=max;
396 if (c <= 0.0)
397 {
398 *hue=0.0;
399 *saturation=0.0;
400 return;
401 }
402 if (max == (QuantumScale*red))
403 {
404 *hue=(QuantumScale*green-QuantumScale*blue)/c;
405 if ((QuantumScale*green) < (QuantumScale*blue))
406 *hue+=6.0;
407 }
408 else
409 if (max == (QuantumScale*green))
410 *hue=2.0+(QuantumScale*blue-QuantumScale*red)/c;
411 else
412 *hue=4.0+(QuantumScale*red-QuantumScale*green)/c;
413 *hue*=60.0/360.0;
414 *saturation=c/max;
415}
416
417static inline void ConvertXYZToLCHab(const double X,const double Y,
418 const double Z,double *luma,double *chroma,double *hue)
419{
420 double
421 a,
422 b;
423
424 ConvertXYZToLab(X,Y,Z,luma,&a,&b);
cristy846740b2013-06-19 18:19:55 +0000425 *chroma=hypot(255.0*(a-0.5),255.0*(b-0.5))/255.0+0.5;
cristy6d897c72013-06-19 19:20:40 +0000426 *hue=180.0*atan2(255.0*(b-0.5),255.0*(a-0.5))/MagickPI/360.0;
cristy8d24e062013-05-07 12:48:19 +0000427 if (*hue < 0.0)
428 *hue+=1.0;
429}
430
431static void ConvertRGBToLCHab(const double red,const double green,
432 const double blue,double *luma,double *chroma,double *hue)
433{
434 double
435 X,
436 Y,
437 Z;
438
439 ConvertRGBToXYZ(red,green,blue,&X,&Y,&Z);
440 ConvertXYZToLCHab(X,Y,Z,luma,chroma,hue);
441}
442
443static inline void ConvertLMSToXYZ(const double L,const double M,const double S,
444 double *X,double *Y,double *Z)
445{
446 *X=1.096123820835514*L-0.278869000218287*M+0.182745179382773*S;
447 *Y=0.454369041975359*L+0.473533154307412*M+0.072097803717229*S;
448 *Z=(-0.009627608738429)*L-0.005698031216113*M+1.015325639954543*S;
449}
450
451static inline void ConvertLMSToRGB(const double L,const double M,
452 const double S,double *red,double *green,double *blue)
453{
454 double
455 X,
456 Y,
457 Z;
458
459 ConvertLMSToXYZ(L,M,S,&X,&Y,&Z);
460 ConvertXYZToRGB(X,Y,Z,red,green,blue);
461}
462
463static inline void ConvertXYZToLMS(const double x,const double y,
464 const double z,double *L,double *M,double *S)
465{
cristy8d24e062013-05-07 12:48:19 +0000466 *L=0.7328*x+0.4296*y-0.1624*z;
467 *M=(-0.7036*x+1.6975*y+0.0061*z);
468 *S=0.0030*x+0.0136*y+0.9834*z;
469}
470
471static void ConvertRGBToLMS(const double red,const double green,
472 const double blue,double *L,double *M,double *S)
473{
474 double
475 X,
476 Y,
477 Z;
478
479 ConvertRGBToXYZ(red,green,blue,&X,&Y,&Z);
480 ConvertXYZToLMS(X,Y,Z,L,M,S);
481}
482
cristy8d24e062013-05-07 12:48:19 +0000483static inline void ConvertXYZToLuv(const double X,const double Y,const double Z,
484 double *L,double *u,double *v)
485{
486 double
487 alpha;
488
489 if ((Y/D65Y) > CIEEpsilon)
490 *L=(double) (116.0*pow(Y/D65Y,1.0/3.0)-16.0);
491 else
492 *L=CIEK*(Y/D65Y);
493 alpha=PerceptibleReciprocal(X+15.0*Y+3.0*Z);
494 *u=13.0*(*L)*((4.0*alpha*X)-(4.0*D65X/(D65X+15.0*D65Y+3.0*D65Z)));
495 *v=13.0*(*L)*((9.0*alpha*Y)-(9.0*D65Y/(D65X+15.0*D65Y+3.0*D65Z)));
496 *L/=100.0;
497 *u=(*u+134.0)/354.0;
498 *v=(*v+140.0)/262.0;
499}
500
501static void ConvertRGBToLuv(const double red,const double green,
502 const double blue,double *L,double *u,double *v)
503{
504 double
505 X,
506 Y,
507 Z;
508
509 ConvertRGBToXYZ(red,green,blue,&X,&Y,&Z);
510 ConvertXYZToLuv(X,Y,Z,L,u,v);
511}
512
513static inline void ConvertLuvToXYZ(const double L,const double u,const double v,
514 double *X,double *Y,double *Z)
515{
516 if (L > (CIEK*CIEEpsilon))
517 *Y=(double) pow((L+16.0)/116.0,3.0);
518 else
519 *Y=L/CIEK;
520 *X=((*Y*((39.0*L/(v+13.0*L*(9.0*D65Y/(D65X+15.0*D65Y+3.0*D65Z))))-5.0))+
521 5.0*(*Y))/((((52.0f*L/(u+13.0*L*(4.0*D65X/(D65X+15.0*D65Y+3.0*D65Z))))-1.0)/
522 3.0)-(-1.0/3.0));
523 *Z=(*X*(((52.0f*L/(u+13.0*L*(4.0*D65X/(D65X+15.0*D65Y+3.0*D65Z))))-1.0)/3.0))-
524 5.0*(*Y);
525}
526
527static inline void ConvertLuvToRGB(const double L,const double u,
528 const double v,double *red,double *green,double *blue)
529{
530 double
531 X,
532 Y,
533 Z;
534
535 ConvertLuvToXYZ(100.0*L,354.0*u-134.0,262.0*v-140.0,&X,&Y,&Z);
536 ConvertXYZToRGB(X,Y,Z,red,green,blue);
537}
538
539static void ConvertRGBToYDbDr(const double red,const double green,
540 const double blue,double *Y,double *Db,double *Dr)
541{
cristy8d24e062013-05-07 12:48:19 +0000542 *Y=QuantumScale*(0.298839*red+0.586811*green+0.114350*blue);
543 *Db=QuantumScale*(-0.450*red-0.883*green+1.333*blue)+0.5;
544 *Dr=QuantumScale*(-1.333*red+1.116*green+0.217*blue)+0.5;
545}
546
547static void ConvertYDbDrToRGB(const double Y,const double Db,const double Dr,
548 double *red,double *green,double *blue)
549{
cristy8d24e062013-05-07 12:48:19 +0000550 *red=QuantumRange*(Y+9.2303716147657e-05*(Db-0.5)-0.52591263066186533*
551 (Dr-0.5));
552 *green=QuantumRange*(Y-0.12913289889050927*(Db-0.5)+0.26789932820759876*
553 (Dr-0.5));
554 *blue=QuantumRange*(Y+0.66467905997895482*(Db-0.5)-7.9202543533108e-05*
555 (Dr-0.5));
556}
557
558static void ConvertRGBToYIQ(const double red,const double green,
559 const double blue,double *Y,double *I,double *Q)
560{
cristy8d24e062013-05-07 12:48:19 +0000561 *Y=QuantumScale*(0.298839*red+0.586811*green+0.114350*blue);
562 *I=QuantumScale*(0.595716*red-0.274453*green-0.321263*blue)+0.5;
563 *Q=QuantumScale*(0.211456*red-0.522591*green+0.311135*blue)+0.5;
564}
565
566static void ConvertYIQToRGB(const double Y,const double I,const double Q,
567 double *red,double *green,double *blue)
568{
cristy8d24e062013-05-07 12:48:19 +0000569 *red=QuantumRange*(Y+0.9562957197589482261*(I-0.5)+0.6210244164652610754*
570 (Q-0.5));
571 *green=QuantumRange*(Y-0.2721220993185104464*(I-0.5)-0.6473805968256950427*
572 (Q-0.5));
573 *blue=QuantumRange*(Y-1.1069890167364901945*(I-0.5)+1.7046149983646481374*
574 (Q-0.5));
575}
576
577static void ConvertRGBToYUV(const double red,const double green,
578 const double blue,double *Y,double *U,double *V)
579{
cristy8d24e062013-05-07 12:48:19 +0000580 *Y=QuantumScale*(0.298839*red+0.586811*green+0.114350*blue);
581 *U=QuantumScale*((-0.147)*red-0.289*green+0.436*blue)+0.5;
582 *V=QuantumScale*(0.615*red-0.515*green-0.100*blue)+0.5;
583}
584
585static void ConvertYUVToRGB(const double Y,const double U,const double V,
586 double *red,double *green,double *blue)
587{
cristy8d24e062013-05-07 12:48:19 +0000588 *red=QuantumRange*(Y-3.945707070708279e-05*(U-0.5)+1.1398279671717170825*
589 (V-0.5));
590 *green=QuantumRange*(Y-0.3946101641414141437*(U-0.5)-0.5805003156565656797*
591 (V-0.5));
592 *blue=QuantumRange*(Y+2.0319996843434342537*(U-0.5)-4.813762626262513e-04*
593 (V-0.5));
594}
595
596static MagickBooleanType ValidateHSIToRGB()
597{
598 double
599 r,
600 g,
601 b;
602
cristyb8588ab2013-05-07 13:17:38 +0000603 (void) FormatLocaleFile(stdout," HSIToRGB");
cristy8d24e062013-05-07 12:48:19 +0000604 ConvertHSIToRGB(111.244375/360.0,0.295985,0.658734,&r,&g,&b);
605 if ((fabs(r-0.545877*QuantumRange) >= ReferenceEpsilon) ||
606 (fabs(g-0.966567*QuantumRange) >= ReferenceEpsilon) ||
607 (fabs(b-0.463759*QuantumRange) >= ReferenceEpsilon))
608 return(MagickFalse);
609 return(MagickTrue);
610}
611
612static MagickBooleanType ValidateRGBToHSI()
613{
614 double
615 h,
616 i,
617 s;
618
cristyb8588ab2013-05-07 13:17:38 +0000619 (void) FormatLocaleFile(stdout," RGBToHSI");
cristy8d24e062013-05-07 12:48:19 +0000620 ConvertRGBToHSI(0.545877*QuantumRange,0.966567*QuantumRange,
621 0.463759*QuantumRange,&h,&s,&i);
622 if ((fabs(h-111.244374/360.0) >= ReferenceEpsilon) ||
623 (fabs(s-0.295985) >= ReferenceEpsilon) ||
624 (fabs(i-0.658734) >= ReferenceEpsilon))
625 return(MagickFalse);
626 return(MagickTrue);
627}
628
629static MagickBooleanType ValidateHSLToRGB()
630{
631 double
632 r,
633 g,
634 b;
635
cristyb8588ab2013-05-07 13:17:38 +0000636 (void) FormatLocaleFile(stdout," HSLToRGB");
cristy8d24e062013-05-07 12:48:19 +0000637 ConvertHSLToRGB(110.200859/360.0,0.882623,0.715163,&r,&g,&b);
638 if ((fabs(r-0.545877*QuantumRange) >= ReferenceEpsilon) ||
639 (fabs(g-0.966567*QuantumRange) >= ReferenceEpsilon) ||
640 (fabs(b-0.463759*QuantumRange) >= ReferenceEpsilon))
641 return(MagickFalse);
642 return(MagickTrue);
643}
644
645static MagickBooleanType ValidateRGBToHSL()
646{
647 double
648 h,
649 l,
650 s;
651
cristyb8588ab2013-05-07 13:17:38 +0000652 (void) FormatLocaleFile(stdout," RGBToHSL");
cristy8d24e062013-05-07 12:48:19 +0000653 ConvertRGBToHSL(0.545877*QuantumRange,0.966567*QuantumRange,
654 0.463759*QuantumRange,&h,&s,&l);
655 if ((fabs(h-110.200859/360.0) >= ReferenceEpsilon) ||
656 (fabs(s-0.882623) >= ReferenceEpsilon) ||
657 (fabs(l-0.715163) >= ReferenceEpsilon))
658 return(MagickFalse);
659 return(MagickTrue);
660}
661
662static MagickBooleanType ValidateHSVToRGB()
663{
664 double
665 r,
666 g,
667 b;
668
cristyb8588ab2013-05-07 13:17:38 +0000669 (void) FormatLocaleFile(stdout," HSVToRGB");
cristy8d24e062013-05-07 12:48:19 +0000670 ConvertHSVToRGB(110.200859/360.0,0.520200,0.966567,&r,&g,&b);
671 if ((fabs(r-0.545877*QuantumRange) >= ReferenceEpsilon) ||
672 (fabs(g-0.966567*QuantumRange) >= ReferenceEpsilon) ||
673 (fabs(b-0.463759*QuantumRange) >= ReferenceEpsilon))
674 return(MagickFalse);
675 return(MagickTrue);
676}
677
678static MagickBooleanType ValidateRGBToHSV()
679{
680 double
681 h,
682 s,
683 v;
684
cristyb8588ab2013-05-07 13:17:38 +0000685 (void) FormatLocaleFile(stdout," RGBToHSV");
cristy8d24e062013-05-07 12:48:19 +0000686 ConvertRGBToHSV(0.545877*QuantumRange,0.966567*QuantumRange,
687 0.463759*QuantumRange,&h,&s,&v);
688 if ((fabs(h-110.200859/360.0) >= ReferenceEpsilon) ||
689 (fabs(s-0.520200) >= ReferenceEpsilon) ||
690 (fabs(v-0.966567) >= ReferenceEpsilon))
691 return(MagickFalse);
692 return(MagickTrue);
693}
694
695static MagickBooleanType ValidateRGBToJPEGYCbCr()
696{
697 double
698 Cb,
699 Cr,
700 Y;
701
cristyb8588ab2013-05-07 13:17:38 +0000702 (void) FormatLocaleFile(stdout," RGBToJPEGYCbCr");
cristy8d24e062013-05-07 12:48:19 +0000703 ConvertRGBToYCbCr(0.545877*QuantumRange,0.966567*QuantumRange,
704 0.463759*QuantumRange,&Y,&Cb,&Cr);
705 if ((fabs(Y-0.783460) >= ReferenceEpsilon) ||
706 (fabs(Cb-0.319581) >= ReferenceEpsilon) ||
707 (fabs(Cr-0.330539) >= ReferenceEpsilon))
708 return(MagickFalse);
709 return(MagickTrue);
710}
711
712static MagickBooleanType ValidateJPEGYCbCrToRGB()
713{
714 double
715 r,
716 g,
717 b;
718
cristyb8588ab2013-05-07 13:17:38 +0000719 (void) FormatLocaleFile(stdout," JPEGYCbCrToRGB");
cristy8d24e062013-05-07 12:48:19 +0000720 ConvertYCbCrToRGB(0.783460,0.319581,0.330539,&r,&g,&b);
721 if ((fabs(r-0.545877*QuantumRange) >= ReferenceEpsilon) ||
722 (fabs(g-0.966567*QuantumRange) >= ReferenceEpsilon) ||
723 (fabs(b-0.463759*QuantumRange) >= ReferenceEpsilon))
724 return(MagickFalse);
725 return(MagickTrue);
726}
727
728static MagickBooleanType ValidateLabToRGB()
729{
730 double
731 r,
732 g,
733 b;
734
cristyb8588ab2013-05-07 13:17:38 +0000735 (void) FormatLocaleFile(stdout," LabToRGB");
cristy8d24e062013-05-07 12:48:19 +0000736 ConvertLabToRGB(88.456154/100.0,-54.671483/255+0.5,51.662818/255.0+0.5,
737 &r,&g,&b);
738 if ((fabs(r-0.545877*QuantumRange) >= ReferenceEpsilon) ||
739 (fabs(g-0.966567*QuantumRange) >= ReferenceEpsilon) ||
740 (fabs(b-0.463759*QuantumRange) >= ReferenceEpsilon))
741 return(MagickFalse);
742 return(MagickTrue);
743}
744
745static MagickBooleanType ValidateRGBToLab()
746{
747 double
748 a,
749 b,
750 L;
751
cristyb8588ab2013-05-07 13:17:38 +0000752 (void) FormatLocaleFile(stdout," RGBToLab");
cristy8d24e062013-05-07 12:48:19 +0000753 ConvertRGBToLab(0.545877*QuantumRange,0.966567*QuantumRange,
754 0.463759*QuantumRange,&L,&a,&b);
755 if ((fabs(L-(88.456154/100.0)) >= ReferenceEpsilon) ||
756 (fabs(a-(-54.671483/255.0+0.5)) >= ReferenceEpsilon) ||
757 (fabs(b-(51.662818/255.0+0.5)) >= ReferenceEpsilon))
758 return(MagickFalse);
759 return(MagickTrue);
760}
761
762static MagickBooleanType ValidateLchToRGB()
763{
764 double
765 b,
766 g,
767 r;
768
cristyb8588ab2013-05-07 13:17:38 +0000769 (void) FormatLocaleFile(stdout," LchToRGB");
cristya44ed132013-06-19 23:56:32 +0000770 ConvertLCHabToRGB(88.456154/100.0,75.219797/255.0+0.5,136.620717/360.0,
cristy8d24e062013-05-07 12:48:19 +0000771 &r,&g,&b);
772 if ((fabs(r-0.545877*QuantumRange) >= ReferenceEpsilon) ||
773 (fabs(g-0.966567*QuantumRange) >= ReferenceEpsilon) ||
774 (fabs(b-0.463759*QuantumRange) >= ReferenceEpsilon))
775 return(MagickFalse);
776 return(MagickTrue);
777}
778
779static MagickBooleanType ValidateRGBToLch()
780{
781 double
782 c,
783 h,
784 L;
785
cristyb8588ab2013-05-07 13:17:38 +0000786 (void) FormatLocaleFile(stdout," RGBToLch");
cristy8d24e062013-05-07 12:48:19 +0000787 ConvertRGBToLCHab(0.545877*QuantumRange,0.966567*QuantumRange,
788 0.463759*QuantumRange,&L,&c,&h);
789 if ((fabs(L-88.456154/100.0) >= ReferenceEpsilon) ||
790 (fabs(c-(75.219797/255.0+0.5)) >= ReferenceEpsilon) ||
791 (fabs(h-(136.620717/255.0+0.5)) >= ReferenceEpsilon))
792 return(MagickFalse);
793 return(MagickTrue);
794}
795
796static MagickBooleanType ValidateRGBToLMS()
797{
798 double
799 L,
800 M,
801 S;
802
cristyb8588ab2013-05-07 13:17:38 +0000803 (void) FormatLocaleFile(stdout," RGBToLMS");
cristy8d24e062013-05-07 12:48:19 +0000804 ConvertRGBToLMS(0.545877*QuantumRange,0.966567*QuantumRange,
805 0.463759*QuantumRange,&L,&M,&S);
806 if ((fabs(L-0.611749) >= ReferenceEpsilon) ||
807 (fabs(M-0.910088) >= ReferenceEpsilon) ||
808 (fabs(S-0.294880) >= ReferenceEpsilon))
809 return(MagickFalse);
810 return(MagickTrue);
811}
812
813static MagickBooleanType ValidateLMSToRGB()
814{
815 double
816 r,
817 g,
818 b;
819
cristyb8588ab2013-05-07 13:17:38 +0000820 (void) FormatLocaleFile(stdout," LMSToRGB");
cristy8d24e062013-05-07 12:48:19 +0000821 ConvertLMSToRGB(0.611749,0.910088,0.294880,&r,&g,&b);
822 if ((fabs(r-0.545877*QuantumRange) >= ReferenceEpsilon) ||
823 (fabs(g-0.966567*QuantumRange) >= ReferenceEpsilon) ||
824 (fabs(b-0.463759*QuantumRange) >= ReferenceEpsilon))
825 return(MagickFalse);
826 return(MagickTrue);
827}
828
829static MagickBooleanType ValidateRGBToLuv()
830{
831 double
832 l,
833 u,
834 v;
835
cristyb8588ab2013-05-07 13:17:38 +0000836 (void) FormatLocaleFile(stdout," RGBToLuv");
cristy8d24e062013-05-07 12:48:19 +0000837 ConvertRGBToLuv(0.545877*QuantumRange,0.966567*QuantumRange,
838 0.463759*QuantumRange,&l,&u,&v);
839 if ((fabs(l-88.456154/262.0) >= ReferenceEpsilon) ||
840 (fabs(u-(-51.330414+134.0)/354.0) >= ReferenceEpsilon) ||
841 (fabs(v-(76.405526+140.0)/262.0) >= ReferenceEpsilon))
842 return(MagickFalse);
843 return(MagickTrue);
844}
845
846static MagickBooleanType ValidateLuvToRGB()
847{
848 double
849 r,
850 g,
851 b;
852
cristyb8588ab2013-05-07 13:17:38 +0000853 (void) FormatLocaleFile(stdout," LuvToRGB");
cristy8d24e062013-05-07 12:48:19 +0000854 ConvertLuvToRGB(88.456154/100.0,(-51.330414+134.0)/354.0,
855 (76.405526+140.0)/262.0,&r,&g,&b);
856 if ((fabs(r-0.545877*QuantumRange) >= ReferenceEpsilon) ||
857 (fabs(g-0.966567*QuantumRange) >= ReferenceEpsilon) ||
858 (fabs(b-0.463759*QuantumRange) >= ReferenceEpsilon))
859 return(MagickFalse);
860 return(MagickTrue);
861}
862
863static MagickBooleanType ValidateRGBToXYZ()
864{
865 double
866 x,
867 y,
868 z;
869
cristyb8588ab2013-05-07 13:17:38 +0000870 (void) FormatLocaleFile(stdout," RGBToXYZ");
cristy8d24e062013-05-07 12:48:19 +0000871 ConvertRGBToXYZ(0.545877*QuantumRange,0.966567*QuantumRange,
872 0.463759*QuantumRange,&x,&y,&z);
873 if ((fabs(x-0.470646) >= ReferenceEpsilon) ||
874 (fabs(y-0.730178) >= ReferenceEpsilon) ||
875 (fabs(z-0.288324) >= ReferenceEpsilon))
876 return(MagickFalse);
877 return(MagickTrue);
878}
879
880static MagickBooleanType ValidateXYZToRGB()
881{
882 double
883 r,
884 g,
885 b;
886
cristyb8588ab2013-05-07 13:17:38 +0000887 (void) FormatLocaleFile(stdout," XYZToRGB");
cristy8d24e062013-05-07 12:48:19 +0000888 ConvertXYZToRGB(0.470646,0.730178,0.288324,&r,&g,&b);
889 if ((fabs(r-0.545877*QuantumRange) >= ReferenceEpsilon) ||
890 (fabs(g-0.966567*QuantumRange) >= ReferenceEpsilon) ||
891 (fabs(b-0.463759*QuantumRange) >= ReferenceEpsilon))
892 return(MagickFalse);
893 return(MagickTrue);
894}
895
896static MagickBooleanType ValidateYDbDrToRGB()
897{
898 double
899 r,
900 g,
901 b;
902
cristyb8588ab2013-05-07 13:17:38 +0000903 (void) FormatLocaleFile(stdout," YDbDrToRGB");
cristy8d24e062013-05-07 12:48:19 +0000904 ConvertYDbDrToRGB(0.783460,-0.480932+0.5,0.451670+0.5,&r,&g,&b);
905 if ((fabs(r-0.545877*QuantumRange) >= ReferenceEpsilon) ||
906 (fabs(g-0.966567*QuantumRange) >= ReferenceEpsilon) ||
907 (fabs(b-0.463759*QuantumRange) >= ReferenceEpsilon))
908 return(MagickFalse);
909 return(MagickTrue);
910}
911
912static MagickBooleanType ValidateRGBToYDbDr()
913{
914 double
915 Db,
916 Dr,
917 Y;
918
cristyb8588ab2013-05-07 13:17:38 +0000919 (void) FormatLocaleFile(stdout," RGBToYDbDr");
cristy8d24e062013-05-07 12:48:19 +0000920 ConvertRGBToYDbDr(0.545877*QuantumRange,0.966567*QuantumRange,
921 0.463759*QuantumRange,&Y,&Db,&Dr);
922 if ((fabs(Y-0.783460) >= ReferenceEpsilon) ||
923 (fabs(Db-(-0.480932)) >= ReferenceEpsilon) ||
924 (fabs(Dr-0.451670) >= ReferenceEpsilon))
925 return(MagickFalse);
926 return(MagickTrue);
927}
928
929static MagickBooleanType ValidateRGBToYIQ()
930{
931 double
932 i,
933 q,
934 y;
935
cristyb8588ab2013-05-07 13:17:38 +0000936 (void) FormatLocaleFile(stdout," RGBToYIQ");
cristy8d24e062013-05-07 12:48:19 +0000937 ConvertRGBToYIQ(0.545877*QuantumRange,0.966567*QuantumRange,
938 0.463759*QuantumRange,&y,&i,&q);
939 if ((fabs(y-0.783460) >= ReferenceEpsilon) ||
940 (fabs(i-(-0.089078)) >= ReferenceEpsilon) ||
941 (fabs(q-(-0.245399)) >= ReferenceEpsilon))
942 return(MagickFalse);
943 return(MagickTrue);
944}
945
946static MagickBooleanType ValidateYIQToRGB()
947{
948 double
949 r,
950 g,
951 b;
952
cristyb8588ab2013-05-07 13:17:38 +0000953 (void) FormatLocaleFile(stdout," YIQToRGB");
cristy8d24e062013-05-07 12:48:19 +0000954 ConvertYIQToRGB(0.783460,-0.089078+0.5,-0.245399+0.5,&r,&g,&b);
955 if ((fabs(r-0.545877*QuantumRange) >= ReferenceEpsilon) ||
956 (fabs(g-0.966567*QuantumRange) >= ReferenceEpsilon) ||
957 (fabs(b-0.463759*QuantumRange) >= ReferenceEpsilon))
958 return(MagickFalse);
959 return(MagickTrue);
960}
961
962static MagickBooleanType ValidateRGBToYPbPr()
963{
964 double
cristy4ae2e762013-05-07 13:19:24 +0000965 Pb,
966 Pr,
cristy8d24e062013-05-07 12:48:19 +0000967 y;
968
cristyb8588ab2013-05-07 13:17:38 +0000969 (void) FormatLocaleFile(stdout," RGBToYPbPr");
cristy8d24e062013-05-07 12:48:19 +0000970 ConvertRGBToYPbPr(0.545877*QuantumRange,0.966567*QuantumRange,
cristy4ae2e762013-05-07 13:19:24 +0000971 0.463759*QuantumRange,&y,&Pb,&Pr);
cristy8d24e062013-05-07 12:48:19 +0000972 if ((fabs(y-0.783460) >= ReferenceEpsilon) ||
cristy4ae2e762013-05-07 13:19:24 +0000973 (fabs(Pb-(-0.180419)) >= ReferenceEpsilon) ||
974 (fabs(Pr-(-0.169461)) >= ReferenceEpsilon))
cristy8d24e062013-05-07 12:48:19 +0000975 return(MagickFalse);
976 return(MagickTrue);
977}
978
979static MagickBooleanType ValidateYPbPrToRGB()
980{
981 double
982 r,
983 g,
984 b;
985
cristyb8588ab2013-05-07 13:17:38 +0000986 (void) FormatLocaleFile(stdout," YPbPrToRGB");
cristy8d24e062013-05-07 12:48:19 +0000987 ConvertYPbPrToRGB(0.783460,-0.180419+0.5,-0.169461+0.5,&r,&g,&b);
988 if ((fabs(r-0.545877*QuantumRange) >= ReferenceEpsilon) ||
989 (fabs(g-0.966567*QuantumRange) >= ReferenceEpsilon) ||
990 (fabs(b-0.463759*QuantumRange) >= ReferenceEpsilon))
991 return(MagickFalse);
992 return(MagickTrue);
993}
994
995static MagickBooleanType ValidateRGBToYUV()
996{
997 double
998 U,
999 V,
1000 Y;
1001
cristyb8588ab2013-05-07 13:17:38 +00001002 (void) FormatLocaleFile(stdout," RGBToYUV");
cristy8d24e062013-05-07 12:48:19 +00001003 ConvertRGBToYUV(0.545877*QuantumRange,0.966567*QuantumRange,
1004 0.463759*QuantumRange,&Y,&U,&V);
1005 if ((fabs(Y-0.783460) >= ReferenceEpsilon) ||
1006 (fabs(U-(-0.157383)) >= ReferenceEpsilon) ||
1007 (fabs(V-(-0.208443)) >= ReferenceEpsilon))
1008 return(MagickFalse);
1009 return(MagickTrue);
1010}
1011
1012static MagickBooleanType ValidateYUVToRGB()
1013{
1014 double
1015 r,
1016 g,
1017 b;
1018
cristyb8588ab2013-05-07 13:17:38 +00001019 (void) FormatLocaleFile(stdout," YUVToRGB");
cristy8d24e062013-05-07 12:48:19 +00001020 ConvertYUVToRGB(0.783460,-0.157383+0.5,-0.208443+0.5,&r,&g,&b);
1021 if ((fabs(r-0.545877*QuantumRange) >= ReferenceEpsilon) ||
1022 (fabs(g-0.966567*QuantumRange) >= ReferenceEpsilon) ||
1023 (fabs(b-0.463759*QuantumRange) >= ReferenceEpsilon))
1024 return(MagickFalse);
1025 return(MagickTrue);
1026}
1027
cristy572e0972013-05-07 12:51:28 +00001028static size_t ValidateColorspaces(ImageInfo *image_info,size_t *fail,
cristy80e870e2013-05-07 12:23:02 +00001029 ExceptionInfo *exception)
1030{
1031 MagickBooleanType
1032 status;
1033
cristy80e870e2013-05-07 12:23:02 +00001034 size_t
1035 test;
1036
cristy8d24e062013-05-07 12:48:19 +00001037 /*
cristy92e5cea2013-05-08 12:24:12 +00001038 Reference: https://code.google.com/p/chroma.
cristy8d24e062013-05-07 12:48:19 +00001039
1040 Illuminant = D65
1041 Observer = 2° (1931)
1042
1043 XYZ 0.470645, 0.730177, 0.288323
1044 sRGB 0.545877, 0.966567, 0.463759
1045 CAT02 LMS 0.611749, 0.910088, 0.294880
1046 Y'DbDr 0.783460, -0.480932, 0.451670
1047 Y'IQ 0.783460, -0.089078, -0.245399
1048 Y'PbPr 0.783460, -0.180419, -0.169461
1049 Y'UV 0.783460, -0.157383, -0.208443
1050 JPEG-Y'CbCr 0.783460, 0.319581, 0.330539
1051 L*u*v* 88.456154, -51.330414, 76.405526
1052 L*a*b* 88.456154, -54.671483, 51.662818
1053 L*C*H* 88.456154, 75.219797, 136.620717
1054 HSV 110.200859, 0.520200, 0.966567
1055 HSL 110.200859, 0.882623, 0.715163
1056 HSI 111.244375, 0.295985, 0.658734
1057 Y'CbCr 187.577791, 87.586330, 90.040886
cristy8d24e062013-05-07 12:48:19 +00001058 */
cristy80e870e2013-05-07 12:23:02 +00001059 (void) FormatLocaleFile(stdout,"validate colorspaces:\n");
cristy8d24e062013-05-07 12:48:19 +00001060 for (test=0; test < 26; test++)
1061 {
1062 CatchException(exception);
1063 (void) FormatLocaleFile(stdout," test %.20g: ",(double) test);
1064 switch (test)
cristy80e870e2013-05-07 12:23:02 +00001065 {
cristy8d24e062013-05-07 12:48:19 +00001066 case 0: status=ValidateHSIToRGB(); break;
1067 case 1: status=ValidateRGBToHSI(); break;
1068 case 2: status=ValidateHSLToRGB(); break;
1069 case 3: status=ValidateRGBToHSL(); break;
1070 case 4: status=ValidateHSVToRGB(); break;
1071 case 5: status=ValidateRGBToHSV(); break;
1072 case 6: status=ValidateJPEGYCbCrToRGB(); break;
1073 case 7: status=ValidateRGBToJPEGYCbCr(); break;
1074 case 8: status=ValidateLabToRGB(); break;
1075 case 9: status=ValidateRGBToLab(); break;
1076 case 10: status=ValidateLchToRGB(); break;
1077 case 11: status=ValidateRGBToLch(); break;
1078 case 12: status=ValidateLMSToRGB(); break;
1079 case 13: status=ValidateRGBToLMS(); break;
1080 case 14: status=ValidateLuvToRGB(); break;
1081 case 15: status=ValidateRGBToLuv(); break;
1082 case 16: status=ValidateXYZToRGB(); break;
1083 case 17: status=ValidateRGBToXYZ(); break;
1084 case 18: status=ValidateYDbDrToRGB(); break;
1085 case 19: status=ValidateRGBToYDbDr(); break;
1086 case 20: status=ValidateYIQToRGB(); break;
1087 case 21: status=ValidateRGBToYIQ(); break;
1088 case 22: status=ValidateYPbPrToRGB(); break;
1089 case 23: status=ValidateRGBToYPbPr(); break;
1090 case 24: status=ValidateYUVToRGB(); break;
1091 case 25: status=ValidateRGBToYUV(); break;
1092 default: status=MagickFalse;
cristy80e870e2013-05-07 12:23:02 +00001093 }
cristyb8588ab2013-05-07 13:17:38 +00001094 if (status == MagickFalse)
cristy8d24e062013-05-07 12:48:19 +00001095 {
1096 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
1097 GetMagickModule());
1098 (*fail)++;
1099 continue;
1100 }
1101 (void) FormatLocaleFile(stdout,"... pass.\n");
1102 }
cristy80e870e2013-05-07 12:23:02 +00001103 (void) FormatLocaleFile(stdout,
1104 " summary: %.20g subtests; %.20g passed; %.20g failed.\n",(double) test,
1105 (double) (test-(*fail)),(double) *fail);
1106 return(test);
1107}
1108
1109/*
1110%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1111% %
1112% %
1113% %
cristy3ed852e2009-09-05 21:47:34 +00001114% V a l i d a t e C o m p a r e C o m m a n d %
1115% %
1116% %
1117% %
1118%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1119%
1120% ValidateCompareCommand() validates the ImageMagick compare command line
1121% program and returns the number of validation tests that passed and failed.
1122%
1123% The format of the ValidateCompareCommand method is:
1124%
cristybb503372010-05-27 20:51:26 +00001125% size_t ValidateCompareCommand(ImageInfo *image_info,
cristy3ed852e2009-09-05 21:47:34 +00001126% const char *reference_filename,const char *output_filename,
cristybb503372010-05-27 20:51:26 +00001127% size_t *fail,ExceptionInfo *exception)
cristy3ed852e2009-09-05 21:47:34 +00001128%
1129% A description of each parameter follows:
1130%
1131% o image_info: the image info.
1132%
1133% o reference_filename: the reference image filename.
1134%
1135% o output_filename: the output image filename.
1136%
1137% o fail: return the number of validation tests that pass.
1138%
1139% o exception: return any errors or warnings in this structure.
1140%
1141*/
cristybb503372010-05-27 20:51:26 +00001142static size_t ValidateCompareCommand(ImageInfo *image_info,
cristy0d772ed2012-11-10 14:34:50 +00001143 const char *reference_filename,const char *output_filename,size_t *fail,
1144 ExceptionInfo *exception)
cristy3ed852e2009-09-05 21:47:34 +00001145{
1146 char
1147 **arguments,
cristy151b66d2015-04-15 10:50:31 +00001148 command[MagickPathExtent];
cristy3ed852e2009-09-05 21:47:34 +00001149
1150 int
1151 number_arguments;
1152
1153 MagickBooleanType
1154 status;
1155
cristybb503372010-05-27 20:51:26 +00001156 register ssize_t
cristy3ed852e2009-09-05 21:47:34 +00001157 i,
1158 j;
1159
cristybb503372010-05-27 20:51:26 +00001160 size_t
cristy3ed852e2009-09-05 21:47:34 +00001161 test;
1162
1163 test=0;
cristyb51dff52011-05-19 16:55:47 +00001164 (void) FormatLocaleFile(stdout,"validate compare command line program:\n");
cristy3ed852e2009-09-05 21:47:34 +00001165 for (i=0; compare_options[i] != (char *) NULL; i++)
1166 {
1167 CatchException(exception);
cristyb51dff52011-05-19 16:55:47 +00001168 (void) FormatLocaleFile(stdout," test %.20g: %s",(double) (test++),
cristy6aa47ad2010-05-28 19:32:32 +00001169 compare_options[i]);
cristy151b66d2015-04-15 10:50:31 +00001170 (void) FormatLocaleString(command,MagickPathExtent,"%s %s %s %s",
cristy3ed852e2009-09-05 21:47:34 +00001171 compare_options[i],reference_filename,reference_filename,output_filename);
1172 arguments=StringToArgv(command,&number_arguments);
1173 if (arguments == (char **) NULL)
1174 {
cristy1e604812011-05-19 18:07:50 +00001175 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
1176 GetMagickModule());
Cristye2879852017-08-13 12:53:45 -04001177 (void) LogMagickEvent(ExceptionEvent,GetMagickModule(),"%s",
1178 exception->reason);
cristy3ed852e2009-09-05 21:47:34 +00001179 (*fail)++;
1180 continue;
1181 }
cristy8a9106f2011-07-05 14:39:26 +00001182 status=CompareImagesCommand(image_info,number_arguments,arguments,
cristy3ed852e2009-09-05 21:47:34 +00001183 (char **) NULL,exception);
cristye42f6582012-02-11 17:59:50 +00001184 for (j=0; j < (ssize_t) number_arguments; j++)
cristy3ed852e2009-09-05 21:47:34 +00001185 arguments[j]=DestroyString(arguments[j]);
1186 arguments=(char **) RelinquishMagickMemory(arguments);
cristy35892192014-05-26 12:04:36 +00001187 if (status == MagickFalse)
cristy3ed852e2009-09-05 21:47:34 +00001188 {
cristy1e604812011-05-19 18:07:50 +00001189 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
1190 GetMagickModule());
Cristye2879852017-08-13 12:53:45 -04001191 (void) LogMagickEvent(ExceptionEvent,GetMagickModule(),"%s",
1192 exception->reason);
cristy3ed852e2009-09-05 21:47:34 +00001193 (*fail)++;
1194 continue;
1195 }
cristyb51dff52011-05-19 16:55:47 +00001196 (void) FormatLocaleFile(stdout,"... pass.\n");
cristy3ed852e2009-09-05 21:47:34 +00001197 }
cristyb51dff52011-05-19 16:55:47 +00001198 (void) FormatLocaleFile(stdout,
cristy75a26232010-06-03 17:15:02 +00001199 " summary: %.20g subtests; %.20g passed; %.20g failed.\n",(double) test,
1200 (double) (test-(*fail)),(double) *fail);
cristy3ed852e2009-09-05 21:47:34 +00001201 return(test);
1202}
1203
1204/*
1205%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1206% %
1207% %
1208% %
1209% V a l i d a t e C o m p o s i t e C o m m a n d %
1210% %
1211% %
1212% %
1213%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1214%
1215% ValidateCompositeCommand() validates the ImageMagick composite command line
1216% program and returns the number of validation tests that passed and failed.
1217%
1218% The format of the ValidateCompositeCommand method is:
1219%
cristybb503372010-05-27 20:51:26 +00001220% size_t ValidateCompositeCommand(ImageInfo *image_info,
cristy3ed852e2009-09-05 21:47:34 +00001221% const char *reference_filename,const char *output_filename,
cristybb503372010-05-27 20:51:26 +00001222% size_t *fail,ExceptionInfo *exception)
cristy3ed852e2009-09-05 21:47:34 +00001223%
1224% A description of each parameter follows:
1225%
1226% o image_info: the image info.
1227%
1228% o reference_filename: the reference image filename.
1229%
1230% o output_filename: the output image filename.
1231%
1232% o fail: return the number of validation tests that pass.
1233%
1234% o exception: return any errors or warnings in this structure.
1235%
1236*/
cristybb503372010-05-27 20:51:26 +00001237static size_t ValidateCompositeCommand(ImageInfo *image_info,
cristy0d772ed2012-11-10 14:34:50 +00001238 const char *reference_filename,const char *output_filename,size_t *fail,
1239 ExceptionInfo *exception)
cristy3ed852e2009-09-05 21:47:34 +00001240{
1241 char
1242 **arguments,
cristy151b66d2015-04-15 10:50:31 +00001243 command[MagickPathExtent];
cristy3ed852e2009-09-05 21:47:34 +00001244
1245 int
1246 number_arguments;
1247
1248 MagickBooleanType
1249 status;
1250
cristybb503372010-05-27 20:51:26 +00001251 register ssize_t
cristy3ed852e2009-09-05 21:47:34 +00001252 i,
1253 j;
1254
cristybb503372010-05-27 20:51:26 +00001255 size_t
cristy3ed852e2009-09-05 21:47:34 +00001256 test;
1257
1258 test=0;
cristyb51dff52011-05-19 16:55:47 +00001259 (void) FormatLocaleFile(stdout,"validate composite command line program:\n");
cristy3ed852e2009-09-05 21:47:34 +00001260 for (i=0; composite_options[i] != (char *) NULL; i++)
1261 {
1262 CatchException(exception);
cristyb51dff52011-05-19 16:55:47 +00001263 (void) FormatLocaleFile(stdout," test %.20g: %s",(double) (test++),
cristy6aa47ad2010-05-28 19:32:32 +00001264 composite_options[i]);
cristy151b66d2015-04-15 10:50:31 +00001265 (void) FormatLocaleString(command,MagickPathExtent,"%s %s %s %s",
cristy3ed852e2009-09-05 21:47:34 +00001266 reference_filename,composite_options[i],reference_filename,
1267 output_filename);
1268 arguments=StringToArgv(command,&number_arguments);
1269 if (arguments == (char **) NULL)
1270 {
cristy1e604812011-05-19 18:07:50 +00001271 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
1272 GetMagickModule());
cristy3ed852e2009-09-05 21:47:34 +00001273 (*fail)++;
1274 continue;
1275 }
1276 status=CompositeImageCommand(image_info,number_arguments,arguments,
1277 (char **) NULL,exception);
cristye42f6582012-02-11 17:59:50 +00001278 for (j=0; j < (ssize_t) number_arguments; j++)
cristy3ed852e2009-09-05 21:47:34 +00001279 arguments[j]=DestroyString(arguments[j]);
1280 arguments=(char **) RelinquishMagickMemory(arguments);
cristy35892192014-05-26 12:04:36 +00001281 if (status == MagickFalse)
cristy3ed852e2009-09-05 21:47:34 +00001282 {
cristy1e604812011-05-19 18:07:50 +00001283 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
1284 GetMagickModule());
cristy3ed852e2009-09-05 21:47:34 +00001285 (*fail)++;
1286 continue;
1287 }
cristyb51dff52011-05-19 16:55:47 +00001288 (void) FormatLocaleFile(stdout,"... pass.\n");
cristy3ed852e2009-09-05 21:47:34 +00001289 }
cristyb51dff52011-05-19 16:55:47 +00001290 (void) FormatLocaleFile(stdout,
cristy75a26232010-06-03 17:15:02 +00001291 " summary: %.20g subtests; %.20g passed; %.20g failed.\n",(double) test,
1292 (double) (test-(*fail)),(double) *fail);
cristy3ed852e2009-09-05 21:47:34 +00001293 return(test);
1294}
1295
1296/*
1297%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1298% %
1299% %
1300% %
1301% V a l i d a t e C o n v e r t C o m m a n d %
1302% %
1303% %
1304% %
1305%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1306%
1307% ValidateConvertCommand() validates the ImageMagick convert command line
1308% program and returns the number of validation tests that passed and failed.
1309%
1310% The format of the ValidateConvertCommand method is:
1311%
cristybb503372010-05-27 20:51:26 +00001312% size_t ValidateConvertCommand(ImageInfo *image_info,
cristy3ed852e2009-09-05 21:47:34 +00001313% const char *reference_filename,const char *output_filename,
cristybb503372010-05-27 20:51:26 +00001314% size_t *fail,ExceptionInfo *exception)
cristy3ed852e2009-09-05 21:47:34 +00001315%
1316% A description of each parameter follows:
1317%
1318% o image_info: the image info.
1319%
1320% o reference_filename: the reference image filename.
1321%
1322% o output_filename: the output image filename.
1323%
1324% o fail: return the number of validation tests that pass.
1325%
1326% o exception: return any errors or warnings in this structure.
1327%
1328*/
cristybb503372010-05-27 20:51:26 +00001329static size_t ValidateConvertCommand(ImageInfo *image_info,
cristy0d772ed2012-11-10 14:34:50 +00001330 const char *reference_filename,const char *output_filename,size_t *fail,
1331 ExceptionInfo *exception)
cristy3ed852e2009-09-05 21:47:34 +00001332{
1333 char
1334 **arguments,
cristy151b66d2015-04-15 10:50:31 +00001335 command[MagickPathExtent];
cristy3ed852e2009-09-05 21:47:34 +00001336
1337 int
1338 number_arguments;
1339
1340 MagickBooleanType
1341 status;
1342
cristybb503372010-05-27 20:51:26 +00001343 register ssize_t
cristy3ed852e2009-09-05 21:47:34 +00001344 i,
1345 j;
1346
cristybb503372010-05-27 20:51:26 +00001347 size_t
cristy3ed852e2009-09-05 21:47:34 +00001348 test;
1349
1350 test=0;
cristyb51dff52011-05-19 16:55:47 +00001351 (void) FormatLocaleFile(stdout,"validate convert command line program:\n");
cristy3ed852e2009-09-05 21:47:34 +00001352 for (i=0; convert_options[i] != (char *) NULL; i++)
1353 {
1354 CatchException(exception);
cristyb51dff52011-05-19 16:55:47 +00001355 (void) FormatLocaleFile(stdout," test %.20g: %s",(double) test++,
cristy6aa47ad2010-05-28 19:32:32 +00001356 convert_options[i]);
cristy151b66d2015-04-15 10:50:31 +00001357 (void) FormatLocaleString(command,MagickPathExtent,"%s %s %s %s",
cristy3ed852e2009-09-05 21:47:34 +00001358 reference_filename,convert_options[i],reference_filename,output_filename);
1359 arguments=StringToArgv(command,&number_arguments);
1360 if (arguments == (char **) NULL)
1361 {
cristy1e604812011-05-19 18:07:50 +00001362 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
1363 GetMagickModule());
cristy3ed852e2009-09-05 21:47:34 +00001364 (*fail)++;
1365 continue;
1366 }
1367 status=ConvertImageCommand(image_info,number_arguments,arguments,
1368 (char **) NULL,exception);
cristye42f6582012-02-11 17:59:50 +00001369 for (j=0; j < (ssize_t) number_arguments; j++)
cristy3ed852e2009-09-05 21:47:34 +00001370 arguments[j]=DestroyString(arguments[j]);
1371 arguments=(char **) RelinquishMagickMemory(arguments);
cristy35892192014-05-26 12:04:36 +00001372 if (status == MagickFalse)
cristy3ed852e2009-09-05 21:47:34 +00001373 {
cristy1e604812011-05-19 18:07:50 +00001374 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
1375 GetMagickModule());
cristy3ed852e2009-09-05 21:47:34 +00001376 (*fail)++;
1377 continue;
1378 }
cristyb51dff52011-05-19 16:55:47 +00001379 (void) FormatLocaleFile(stdout,"... pass.\n");
cristy3ed852e2009-09-05 21:47:34 +00001380 }
cristyb51dff52011-05-19 16:55:47 +00001381 (void) FormatLocaleFile(stdout,
cristy75a26232010-06-03 17:15:02 +00001382 " summary: %.20g subtests; %.20g passed; %.20g failed.\n",(double) test,
1383 (double) (test-(*fail)),(double) *fail);
cristy3ed852e2009-09-05 21:47:34 +00001384 return(test);
1385}
1386
1387/*
1388%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1389% %
1390% %
1391% %
1392% V a l i d a t e I d e n t i f y C o m m a n d %
1393% %
1394% %
1395% %
1396%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1397%
1398% ValidateIdentifyCommand() validates the ImageMagick identify command line
1399% program and returns the number of validation tests that passed and failed.
1400%
1401% The format of the ValidateIdentifyCommand method is:
1402%
cristybb503372010-05-27 20:51:26 +00001403% size_t ValidateIdentifyCommand(ImageInfo *image_info,
cristy3ed852e2009-09-05 21:47:34 +00001404% const char *reference_filename,const char *output_filename,
cristybb503372010-05-27 20:51:26 +00001405% size_t *fail,ExceptionInfo *exception)
cristy3ed852e2009-09-05 21:47:34 +00001406%
1407% A description of each parameter follows:
1408%
1409% o image_info: the image info.
1410%
1411% o reference_filename: the reference image filename.
1412%
1413% o output_filename: the output image filename.
1414%
1415% o fail: return the number of validation tests that pass.
1416%
1417% o exception: return any errors or warnings in this structure.
1418%
1419*/
cristybb503372010-05-27 20:51:26 +00001420static size_t ValidateIdentifyCommand(ImageInfo *image_info,
cristy0d772ed2012-11-10 14:34:50 +00001421 const char *reference_filename,const char *output_filename,size_t *fail,
1422 ExceptionInfo *exception)
cristy3ed852e2009-09-05 21:47:34 +00001423{
1424 char
1425 **arguments,
cristy151b66d2015-04-15 10:50:31 +00001426 command[MagickPathExtent];
cristy3ed852e2009-09-05 21:47:34 +00001427
1428 int
1429 number_arguments;
1430
1431 MagickBooleanType
1432 status;
1433
cristybb503372010-05-27 20:51:26 +00001434 register ssize_t
cristy3ed852e2009-09-05 21:47:34 +00001435 i,
1436 j;
1437
cristybb503372010-05-27 20:51:26 +00001438 size_t
cristy3ed852e2009-09-05 21:47:34 +00001439 test;
1440
1441 (void) output_filename;
1442 test=0;
cristyb51dff52011-05-19 16:55:47 +00001443 (void) FormatLocaleFile(stdout,"validate identify command line program:\n");
cristy3ed852e2009-09-05 21:47:34 +00001444 for (i=0; identify_options[i] != (char *) NULL; i++)
1445 {
1446 CatchException(exception);
cristyb51dff52011-05-19 16:55:47 +00001447 (void) FormatLocaleFile(stdout," test %.20g: %s",(double) test++,
cristy6aa47ad2010-05-28 19:32:32 +00001448 identify_options[i]);
cristy151b66d2015-04-15 10:50:31 +00001449 (void) FormatLocaleString(command,MagickPathExtent,"%s %s",
cristy3ed852e2009-09-05 21:47:34 +00001450 identify_options[i],reference_filename);
1451 arguments=StringToArgv(command,&number_arguments);
1452 if (arguments == (char **) NULL)
1453 {
cristy1e604812011-05-19 18:07:50 +00001454 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
1455 GetMagickModule());
cristy3ed852e2009-09-05 21:47:34 +00001456 (*fail)++;
1457 continue;
1458 }
1459 status=IdentifyImageCommand(image_info,number_arguments,arguments,
1460 (char **) NULL,exception);
cristye42f6582012-02-11 17:59:50 +00001461 for (j=0; j < (ssize_t) number_arguments; j++)
cristy3ed852e2009-09-05 21:47:34 +00001462 arguments[j]=DestroyString(arguments[j]);
1463 arguments=(char **) RelinquishMagickMemory(arguments);
cristy35892192014-05-26 12:04:36 +00001464 if (status == MagickFalse)
cristy3ed852e2009-09-05 21:47:34 +00001465 {
cristy1e604812011-05-19 18:07:50 +00001466 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
1467 GetMagickModule());
cristy3ed852e2009-09-05 21:47:34 +00001468 (*fail)++;
1469 continue;
1470 }
cristyb51dff52011-05-19 16:55:47 +00001471 (void) FormatLocaleFile(stdout,"... pass.\n");
cristy3ed852e2009-09-05 21:47:34 +00001472 }
cristyb51dff52011-05-19 16:55:47 +00001473 (void) FormatLocaleFile(stdout,
cristy75a26232010-06-03 17:15:02 +00001474 " summary: %.20g subtests; %.20g passed; %.20g failed.\n",(double) test,
1475 (double) (test-(*fail)),(double) *fail);
cristy3ed852e2009-09-05 21:47:34 +00001476 return(test);
1477}
1478
1479/*
1480%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1481% %
1482% %
1483% %
1484% V a l i d a t e I m a g e F o r m a t s I n M e m o r y %
1485% %
1486% %
1487% %
1488%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1489%
1490% ValidateImageFormatsInMemory() validates the ImageMagick image formats in
1491% memory and returns the number of validation tests that passed and failed.
1492%
1493% The format of the ValidateImageFormatsInMemory method is:
1494%
cristybb503372010-05-27 20:51:26 +00001495% size_t ValidateImageFormatsInMemory(ImageInfo *image_info,
cristy3ed852e2009-09-05 21:47:34 +00001496% const char *reference_filename,const char *output_filename,
cristybb503372010-05-27 20:51:26 +00001497% size_t *fail,ExceptionInfo *exception)
cristy3ed852e2009-09-05 21:47:34 +00001498%
1499% A description of each parameter follows:
1500%
1501% o image_info: the image info.
1502%
1503% o reference_filename: the reference image filename.
1504%
1505% o output_filename: the output image filename.
1506%
1507% o fail: return the number of validation tests that pass.
1508%
1509% o exception: return any errors or warnings in this structure.
1510%
1511*/
glennrp4084cda2012-12-11 14:13:35 +00001512
cristy725833f2012-12-13 17:40:53 +00001513/*
1514 Enable this to count remaining $TMPDIR/magick-* files. Note that the count
1515 includes any files left over from other runs.
1516*/
1517#undef MagickCountTempFiles
glennrp4084cda2012-12-11 14:13:35 +00001518
cristybb503372010-05-27 20:51:26 +00001519static size_t ValidateImageFormatsInMemory(ImageInfo *image_info,
cristy0d772ed2012-11-10 14:34:50 +00001520 const char *reference_filename,const char *output_filename,size_t *fail,
1521 ExceptionInfo *exception)
cristy3ed852e2009-09-05 21:47:34 +00001522{
1523 char
glennrp4084cda2012-12-11 14:13:35 +00001524#ifdef MagickCountTempFiles
cristy151b66d2015-04-15 10:50:31 +00001525 path[MagickPathExtent],
1526 SystemCommand[MagickPathExtent],
glennrp4084cda2012-12-11 14:13:35 +00001527#endif
cristy151b66d2015-04-15 10:50:31 +00001528 size[MagickPathExtent];
cristy3ed852e2009-09-05 21:47:34 +00001529
1530 const MagickInfo
1531 *magick_info;
1532
1533 double
1534 distortion,
1535 fuzz;
1536
1537 Image
1538 *difference_image,
cristy2c404222012-06-10 15:36:30 +00001539 *ping_image,
cristy6b9ae3f2012-06-10 15:39:34 +00001540 *reconstruct_image,
cristy100b8d92012-01-08 00:32:49 +00001541 *reference_image;
cristy3ed852e2009-09-05 21:47:34 +00001542
1543 MagickBooleanType
1544 status;
1545
cristybb503372010-05-27 20:51:26 +00001546 register ssize_t
cristy3ed852e2009-09-05 21:47:34 +00001547 i,
1548 j;
1549
1550 size_t
cristy100b8d92012-01-08 00:32:49 +00001551 length,
1552 test;
cristy3ed852e2009-09-05 21:47:34 +00001553
1554 unsigned char
1555 *blob;
1556
cristy3ed852e2009-09-05 21:47:34 +00001557 test=0;
cristyb51dff52011-05-19 16:55:47 +00001558 (void) FormatLocaleFile(stdout,"validate image formats in memory:\n");
glennrp4084cda2012-12-11 14:13:35 +00001559
1560#ifdef MagickCountTempFiles
1561 (void)GetPathTemplate(path);
glennrp33fb8032013-01-07 18:43:03 +00001562 /* Remove file template except for the leading "/path/to/magick-" */
glennrp4084cda2012-12-11 14:13:35 +00001563 path[strlen(path)-17]='\0';
1564 (void) FormatLocaleFile(stdout," tmp path is '%s*'\n",path);
1565#endif
1566
cristy3ed852e2009-09-05 21:47:34 +00001567 for (i=0; reference_formats[i].magick != (char *) NULL; i++)
1568 {
1569 magick_info=GetMagickInfo(reference_formats[i].magick,exception);
1570 if ((magick_info == (const MagickInfo *) NULL) ||
1571 (magick_info->decoder == (DecodeImageHandler *) NULL) ||
1572 (magick_info->encoder == (EncodeImageHandler *) NULL))
1573 continue;
cristy5f1c1ff2010-12-23 21:38:06 +00001574 for (j=0; reference_types[j].type != UndefinedType; j++)
cristy3ed852e2009-09-05 21:47:34 +00001575 {
1576 /*
1577 Generate reference image.
1578 */
1579 CatchException(exception);
cristy1e604812011-05-19 18:07:50 +00001580 (void) FormatLocaleFile(stdout," test %.20g: %s/%s/%s/%.20g-bits",
1581 (double) (test++),reference_formats[i].magick,CommandOptionToMnemonic(
cristy7998e5f2009-10-08 02:45:33 +00001582 MagickCompressOptions,reference_formats[i].compression),
cristy042ee782011-04-22 18:48:30 +00001583 CommandOptionToMnemonic(MagickTypeOptions,reference_types[j].type),
cristy75a26232010-06-03 17:15:02 +00001584 (double) reference_types[j].depth);
cristy3ed852e2009-09-05 21:47:34 +00001585 (void) CopyMagickString(image_info->filename,reference_filename,
cristy151b66d2015-04-15 10:50:31 +00001586 MagickPathExtent);
cristy3ed852e2009-09-05 21:47:34 +00001587 reference_image=ReadImage(image_info,exception);
Cristye2879852017-08-13 12:53:45 -04001588 if ((reference_image == (Image *) NULL) ||
1589 (exception->severity >= ErrorException))
cristy3ed852e2009-09-05 21:47:34 +00001590 {
cristy1e604812011-05-19 18:07:50 +00001591 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
1592 GetMagickModule());
Cristye2879852017-08-13 12:53:45 -04001593 if (exception->reason != (char *) NULL)
1594 (void) FormatLocaleFile(stdout," reason:%s\n",exception->reason);
dirk775194c2014-05-17 19:47:54 +00001595 CatchException(exception);
cristy3ed852e2009-09-05 21:47:34 +00001596 (*fail)++;
1597 continue;
1598 }
1599 /*
1600 Write reference image.
1601 */
cristy151b66d2015-04-15 10:50:31 +00001602 (void) FormatLocaleString(size,MagickPathExtent,"%.20gx%.20g",
cristy75a26232010-06-03 17:15:02 +00001603 (double) reference_image->columns,(double) reference_image->rows);
cristy3ed852e2009-09-05 21:47:34 +00001604 (void) CloneString(&image_info->size,size);
1605 image_info->depth=reference_types[j].depth;
Cristy114baf52017-07-12 19:21:05 -04001606 (void) FormatLocaleString(reference_image->filename,MagickPathExtent,
1607 "%s:%s",reference_formats[i].magick,output_filename);
cristy018f07f2011-09-04 21:15:19 +00001608 status=SetImageType(reference_image,reference_types[j].type,exception);
Cristye2879852017-08-13 12:53:45 -04001609 if (status == MagickFalse || (exception->severity >= ErrorException))
cristy3ed852e2009-09-05 21:47:34 +00001610 {
cristy1e604812011-05-19 18:07:50 +00001611 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
1612 GetMagickModule());
Cristye2879852017-08-13 12:53:45 -04001613 if (exception->reason != (char *) NULL)
1614 (void) FormatLocaleFile(stdout," reason:%s\n",exception->reason);
dirk775194c2014-05-17 19:47:54 +00001615 CatchException(exception);
cristy3ed852e2009-09-05 21:47:34 +00001616 (*fail)++;
1617 reference_image=DestroyImage(reference_image);
1618 continue;
1619 }
cristy8a11cb12011-10-19 23:53:34 +00001620 status=SetImageDepth(reference_image,reference_types[j].depth,exception);
Cristye2879852017-08-13 12:53:45 -04001621 if (status == MagickFalse || (exception->severity >= ErrorException))
cristy3ed852e2009-09-05 21:47:34 +00001622 {
cristy1e604812011-05-19 18:07:50 +00001623 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
1624 GetMagickModule());
dirk775194c2014-05-17 19:47:54 +00001625 CatchException(exception);
cristy3ed852e2009-09-05 21:47:34 +00001626 (*fail)++;
1627 reference_image=DestroyImage(reference_image);
1628 continue;
1629 }
cristy7998e5f2009-10-08 02:45:33 +00001630 reference_image->compression=reference_formats[i].compression;
cristy6f9e0d32011-08-28 16:32:09 +00001631 status=WriteImage(image_info,reference_image,exception);
cristy3ed852e2009-09-05 21:47:34 +00001632 reference_image=DestroyImage(reference_image);
Cristye2879852017-08-13 12:53:45 -04001633 if (status == MagickFalse || (exception->severity >= ErrorException))
cristy3ed852e2009-09-05 21:47:34 +00001634 {
cristy1e604812011-05-19 18:07:50 +00001635 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
1636 GetMagickModule());
Cristye2879852017-08-13 12:53:45 -04001637 if (exception->reason != (char *) NULL)
1638 (void) FormatLocaleFile(stdout," reason:%s\n",exception->reason);
dirk775194c2014-05-17 19:47:54 +00001639 CatchException(exception);
cristy3ed852e2009-09-05 21:47:34 +00001640 (*fail)++;
1641 continue;
1642 }
1643 /*
cristy2c404222012-06-10 15:36:30 +00001644 Ping reference image.
cristy3ed852e2009-09-05 21:47:34 +00001645 */
cristy151b66d2015-04-15 10:50:31 +00001646 (void) FormatLocaleString(image_info->filename,MagickPathExtent,"%s:%s",
cristy3ed852e2009-09-05 21:47:34 +00001647 reference_formats[i].magick,output_filename);
cristy2c404222012-06-10 15:36:30 +00001648 ping_image=PingImage(image_info,exception);
dirk775194c2014-05-17 19:47:54 +00001649 if (ping_image == (Image *) NULL ||
Cristye2879852017-08-13 12:53:45 -04001650 (exception->severity >= ErrorException))
cristy2c404222012-06-10 15:36:30 +00001651 {
1652 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
1653 GetMagickModule());
Cristye2879852017-08-13 12:53:45 -04001654 if (exception->reason != (char *) NULL)
1655 (void) FormatLocaleFile(stdout," reason:%s\n",exception->reason);
dirk775194c2014-05-17 19:47:54 +00001656 CatchException(exception);
cristy2c404222012-06-10 15:36:30 +00001657 (*fail)++;
1658 continue;
1659 }
1660 ping_image=DestroyImage(ping_image);
1661 /*
1662 Read reference image.
1663 */
cristy3ed852e2009-09-05 21:47:34 +00001664 reference_image=ReadImage(image_info,exception);
Cristye2879852017-08-13 12:53:45 -04001665 if ((reference_image == (Image *) NULL) ||
1666 (exception->severity >= ErrorException))
cristy3ed852e2009-09-05 21:47:34 +00001667 {
cristy1e604812011-05-19 18:07:50 +00001668 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
1669 GetMagickModule());
Cristye2879852017-08-13 12:53:45 -04001670 if (exception->reason != (char *) NULL)
1671 (void) FormatLocaleFile(stdout," reason:%s\n",exception->reason);
dirk775194c2014-05-17 19:47:54 +00001672 CatchException(exception);
cristy3ed852e2009-09-05 21:47:34 +00001673 (*fail)++;
1674 continue;
1675 }
1676 /*
1677 Write reference image.
1678 */
Cristy114baf52017-07-12 19:21:05 -04001679 (void) FormatLocaleString(reference_image->filename,MagickPathExtent,
1680 "%s:%s",reference_formats[i].magick,output_filename);
cristy3ed852e2009-09-05 21:47:34 +00001681 (void) CopyMagickString(image_info->magick,reference_formats[i].magick,
cristy151b66d2015-04-15 10:50:31 +00001682 MagickPathExtent);
cristy3ed852e2009-09-05 21:47:34 +00001683 reference_image->depth=reference_types[j].depth;
cristy7998e5f2009-10-08 02:45:33 +00001684 reference_image->compression=reference_formats[i].compression;
cristy3ed852e2009-09-05 21:47:34 +00001685 length=8192;
1686 blob=ImageToBlob(image_info,reference_image,&length,exception);
Cristye2879852017-08-13 12:53:45 -04001687 if ((blob == (unsigned char *) NULL) ||
1688 (exception->severity >= ErrorException))
cristy3ed852e2009-09-05 21:47:34 +00001689 {
cristy1e604812011-05-19 18:07:50 +00001690 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
1691 GetMagickModule());
Cristye2879852017-08-13 12:53:45 -04001692 if (exception->reason != (char *) NULL)
1693 (void) FormatLocaleFile(stdout," reason:%s\n",exception->reason);
dirk775194c2014-05-17 19:47:54 +00001694 CatchException(exception);
cristy3ed852e2009-09-05 21:47:34 +00001695 (*fail)++;
1696 reference_image=DestroyImage(reference_image);
1697 continue;
1698 }
1699 /*
cristy2c404222012-06-10 15:36:30 +00001700 Ping reference blob.
1701 */
1702 ping_image=PingBlob(image_info,blob,length,exception);
dirk775194c2014-05-17 19:47:54 +00001703 if (ping_image == (Image *) NULL ||
Cristye2879852017-08-13 12:53:45 -04001704 (exception->severity >= ErrorException))
cristy2c404222012-06-10 15:36:30 +00001705 {
1706 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
1707 GetMagickModule());
Cristye2879852017-08-13 12:53:45 -04001708 if (exception->reason != (char *) NULL)
1709 (void) FormatLocaleFile(stdout," reason:%s\n",exception->reason);
dirk775194c2014-05-17 19:47:54 +00001710 CatchException(exception);
cristy2c404222012-06-10 15:36:30 +00001711 (*fail)++;
1712 blob=(unsigned char *) RelinquishMagickMemory(blob);
1713 continue;
1714 }
1715 ping_image=DestroyImage(ping_image);
1716 /*
cristy3ed852e2009-09-05 21:47:34 +00001717 Read reconstruct image.
1718 */
cristy151b66d2015-04-15 10:50:31 +00001719 (void) FormatLocaleString(image_info->filename,MagickPathExtent,"%s:%s",
cristy3ed852e2009-09-05 21:47:34 +00001720 reference_formats[i].magick,output_filename);
1721 reconstruct_image=BlobToImage(image_info,blob,length,exception);
1722 blob=(unsigned char *) RelinquishMagickMemory(blob);
dirk775194c2014-05-17 19:47:54 +00001723 if (reconstruct_image == (Image *) NULL ||
Cristye2879852017-08-13 12:53:45 -04001724 (exception->severity >= ErrorException))
cristy3ed852e2009-09-05 21:47:34 +00001725 {
cristy1e604812011-05-19 18:07:50 +00001726 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
1727 GetMagickModule());
Cristye2879852017-08-13 12:53:45 -04001728 if (exception->reason != (char *) NULL)
1729 (void) FormatLocaleFile(stdout," reason:%s\n",exception->reason);
dirk775194c2014-05-17 19:47:54 +00001730 CatchException(exception);
cristy3ed852e2009-09-05 21:47:34 +00001731 (*fail)++;
1732 reference_image=DestroyImage(reference_image);
1733 continue;
1734 }
1735 /*
1736 Compare reference to reconstruct image.
1737 */
cristy93540ab2013-03-30 19:59:41 +00001738 fuzz=0.003; /* grayscale */
cristy3ed852e2009-09-05 21:47:34 +00001739 if (reference_formats[i].fuzz != 0.0)
1740 fuzz=reference_formats[i].fuzz;
cristy8a9106f2011-07-05 14:39:26 +00001741 difference_image=CompareImages(reference_image,reconstruct_image,
cristy864c6a82012-06-19 16:45:22 +00001742 RootMeanSquaredErrorMetric,&distortion,exception);
cristy3ed852e2009-09-05 21:47:34 +00001743 reconstruct_image=DestroyImage(reconstruct_image);
1744 reference_image=DestroyImage(reference_image);
dirk775194c2014-05-17 19:47:54 +00001745 if (difference_image == (Image *) NULL ||
Cristye2879852017-08-13 12:53:45 -04001746 (exception->severity >= ErrorException))
cristy3ed852e2009-09-05 21:47:34 +00001747 {
cristy1e604812011-05-19 18:07:50 +00001748 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
1749 GetMagickModule());
Cristye2879852017-08-13 12:53:45 -04001750 if (exception->reason != (char *) NULL)
1751 (void) FormatLocaleFile(stdout," reason:%s\n",exception->reason);
dirk775194c2014-05-17 19:47:54 +00001752 CatchException(exception);
cristy3ed852e2009-09-05 21:47:34 +00001753 (*fail)++;
1754 continue;
1755 }
1756 difference_image=DestroyImage(difference_image);
cristy14154222012-06-15 23:21:49 +00001757 if ((QuantumScale*distortion) > fuzz)
cristy3ed852e2009-09-05 21:47:34 +00001758 {
cristyb51dff52011-05-19 16:55:47 +00001759 (void) FormatLocaleFile(stdout,"... fail (with distortion %g).\n",
cristy14154222012-06-15 23:21:49 +00001760 QuantumScale*distortion);
cristy3ed852e2009-09-05 21:47:34 +00001761 (*fail)++;
1762 continue;
1763 }
glennrp4084cda2012-12-11 14:13:35 +00001764#ifdef MagickCountTempFiles
1765 (void) FormatLocaleFile(stdout,"... pass, ");
1766 (void) fflush(stdout);
1767 SystemCommand[0]='\0';
cristy77b33ba2013-03-28 13:23:13 +00001768 (void) strncat(SystemCommand,"echo `ls ",9);
cristy151b66d2015-04-15 10:50:31 +00001769 (void) strncat(SystemCommand,path,MagickPathExtent-31);
cristy77b33ba2013-03-28 13:23:13 +00001770 (void) strncat(SystemCommand,"* | wc -w` tmp files.",20);
1771 (void) system(SystemCommand);
glennrp4084cda2012-12-11 14:13:35 +00001772 (void) fflush(stdout);
1773#else
1774 (void) FormatLocaleFile(stdout,"... pass\n");
1775#endif
cristy3ed852e2009-09-05 21:47:34 +00001776 }
1777 }
cristyb51dff52011-05-19 16:55:47 +00001778 (void) FormatLocaleFile(stdout,
cristy75a26232010-06-03 17:15:02 +00001779 " summary: %.20g subtests; %.20g passed; %.20g failed.\n",(double) test,
1780 (double) (test-(*fail)),(double) *fail);
cristy3ed852e2009-09-05 21:47:34 +00001781 return(test);
1782}
1783
1784/*
1785%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1786% %
1787% %
1788% %
1789% V a l i d a t e I m a g e F o r m a t s O n D i s k %
1790% %
1791% %
1792% %
1793%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1794%
1795% ValidateImageFormatsOnDisk() validates the ImageMagick image formats on disk
1796% and returns the number of validation tests that passed and failed.
1797%
1798% The format of the ValidateImageFormatsOnDisk method is:
1799%
cristybb503372010-05-27 20:51:26 +00001800% size_t ValidateImageFormatsOnDisk(ImageInfo *image_info,
cristy3ed852e2009-09-05 21:47:34 +00001801% const char *reference_filename,const char *output_filename,
cristybb503372010-05-27 20:51:26 +00001802% size_t *fail,ExceptionInfo *exception)
cristy3ed852e2009-09-05 21:47:34 +00001803%
1804% A description of each parameter follows:
1805%
1806% o image_info: the image info.
1807%
1808% o reference_filename: the reference image filename.
1809%
1810% o output_filename: the output image filename.
1811%
1812% o fail: return the number of validation tests that pass.
1813%
1814% o exception: return any errors or warnings in this structure.
1815%
1816*/
cristybb503372010-05-27 20:51:26 +00001817static size_t ValidateImageFormatsOnDisk(ImageInfo *image_info,
cristy0d772ed2012-11-10 14:34:50 +00001818 const char *reference_filename,const char *output_filename,size_t *fail,
1819 ExceptionInfo *exception)
cristy3ed852e2009-09-05 21:47:34 +00001820{
1821 char
cristy151b66d2015-04-15 10:50:31 +00001822 size[MagickPathExtent];
cristy3ed852e2009-09-05 21:47:34 +00001823
1824 const MagickInfo
1825 *magick_info;
1826
1827 double
1828 distortion,
1829 fuzz;
1830
1831 Image
1832 *difference_image,
1833 *reference_image,
1834 *reconstruct_image;
1835
1836 MagickBooleanType
1837 status;
1838
cristybb503372010-05-27 20:51:26 +00001839 register ssize_t
cristy3ed852e2009-09-05 21:47:34 +00001840 i,
1841 j;
1842
cristybb503372010-05-27 20:51:26 +00001843 size_t
cristy3ed852e2009-09-05 21:47:34 +00001844 test;
1845
1846 test=0;
cristyb51dff52011-05-19 16:55:47 +00001847 (void) FormatLocaleFile(stdout,"validate image formats on disk:\n");
cristy3ed852e2009-09-05 21:47:34 +00001848 for (i=0; reference_formats[i].magick != (char *) NULL; i++)
1849 {
1850 magick_info=GetMagickInfo(reference_formats[i].magick,exception);
1851 if ((magick_info == (const MagickInfo *) NULL) ||
1852 (magick_info->decoder == (DecodeImageHandler *) NULL) ||
1853 (magick_info->encoder == (EncodeImageHandler *) NULL))
1854 continue;
cristy5f1c1ff2010-12-23 21:38:06 +00001855 for (j=0; reference_types[j].type != UndefinedType; j++)
cristy3ed852e2009-09-05 21:47:34 +00001856 {
1857 /*
1858 Generate reference image.
1859 */
1860 CatchException(exception);
cristy1e604812011-05-19 18:07:50 +00001861 (void) FormatLocaleFile(stdout," test %.20g: %s/%s/%s/%.20g-bits",
1862 (double) (test++),reference_formats[i].magick,CommandOptionToMnemonic(
cristy7998e5f2009-10-08 02:45:33 +00001863 MagickCompressOptions,reference_formats[i].compression),
cristy042ee782011-04-22 18:48:30 +00001864 CommandOptionToMnemonic(MagickTypeOptions,reference_types[j].type),
cristy75a26232010-06-03 17:15:02 +00001865 (double) reference_types[j].depth);
cristy3ed852e2009-09-05 21:47:34 +00001866 (void) CopyMagickString(image_info->filename,reference_filename,
cristy151b66d2015-04-15 10:50:31 +00001867 MagickPathExtent);
cristy3ed852e2009-09-05 21:47:34 +00001868 reference_image=ReadImage(image_info,exception);
Cristye2879852017-08-13 12:53:45 -04001869 if ((reference_image == (Image *) NULL) ||
1870 (exception->severity >= ErrorException))
cristy3ed852e2009-09-05 21:47:34 +00001871 {
cristy1e604812011-05-19 18:07:50 +00001872 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
1873 GetMagickModule());
Cristye2879852017-08-13 12:53:45 -04001874 if (exception->reason != (char *) NULL)
1875 (void) FormatLocaleFile(stdout," reason:%s\n",exception->reason);
dirk775194c2014-05-17 19:47:54 +00001876 CatchException(exception);
cristy3ed852e2009-09-05 21:47:34 +00001877 (*fail)++;
1878 continue;
1879 }
1880 /*
1881 Write reference image.
1882 */
cristy151b66d2015-04-15 10:50:31 +00001883 (void) FormatLocaleString(size,MagickPathExtent,"%.20gx%.20g",
cristy75a26232010-06-03 17:15:02 +00001884 (double) reference_image->columns,(double) reference_image->rows);
cristy3ed852e2009-09-05 21:47:34 +00001885 (void) CloneString(&image_info->size,size);
1886 image_info->depth=reference_types[j].depth;
Cristy114baf52017-07-12 19:21:05 -04001887 (void) FormatLocaleString(reference_image->filename,MagickPathExtent,
1888 "%s:%s",reference_formats[i].magick,output_filename);
cristy018f07f2011-09-04 21:15:19 +00001889 status=SetImageType(reference_image,reference_types[j].type,exception);
Cristye2879852017-08-13 12:53:45 -04001890 if (status == MagickFalse || (exception->severity >= ErrorException))
cristy3ed852e2009-09-05 21:47:34 +00001891 {
cristy1e604812011-05-19 18:07:50 +00001892 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
1893 GetMagickModule());
Cristye2879852017-08-13 12:53:45 -04001894 if (exception->reason != (char *) NULL)
1895 (void) FormatLocaleFile(stdout," reason:%s\n",exception->reason);
dirk775194c2014-05-17 19:47:54 +00001896 CatchException(exception);
cristy3ed852e2009-09-05 21:47:34 +00001897 (*fail)++;
1898 reference_image=DestroyImage(reference_image);
1899 continue;
1900 }
cristy8a11cb12011-10-19 23:53:34 +00001901 status=SetImageDepth(reference_image,reference_types[j].depth,exception);
Cristye2879852017-08-13 12:53:45 -04001902 if (status == MagickFalse || (exception->severity >= ErrorException))
cristy3ed852e2009-09-05 21:47:34 +00001903 {
cristy1e604812011-05-19 18:07:50 +00001904 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
1905 GetMagickModule());
dirk775194c2014-05-17 19:47:54 +00001906 CatchException(exception);
cristy3ed852e2009-09-05 21:47:34 +00001907 (*fail)++;
1908 reference_image=DestroyImage(reference_image);
1909 continue;
1910 }
cristy7998e5f2009-10-08 02:45:33 +00001911 reference_image->compression=reference_formats[i].compression;
cristy6f9e0d32011-08-28 16:32:09 +00001912 status=WriteImage(image_info,reference_image,exception);
cristy3ed852e2009-09-05 21:47:34 +00001913 reference_image=DestroyImage(reference_image);
Cristye2879852017-08-13 12:53:45 -04001914 if (status == MagickFalse || (exception->severity >= ErrorException))
cristy3ed852e2009-09-05 21:47:34 +00001915 {
cristy1e604812011-05-19 18:07:50 +00001916 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
1917 GetMagickModule());
Cristye2879852017-08-13 12:53:45 -04001918 if (exception->reason != (char *) NULL)
1919 (void) FormatLocaleFile(stdout," reason:%s\n",exception->reason);
dirk775194c2014-05-17 19:47:54 +00001920 CatchException(exception);
cristy3ed852e2009-09-05 21:47:34 +00001921 (*fail)++;
1922 continue;
1923 }
1924 /*
1925 Read reference image.
1926 */
cristy151b66d2015-04-15 10:50:31 +00001927 (void) FormatLocaleString(image_info->filename,MagickPathExtent,"%s:%s",
cristy3ed852e2009-09-05 21:47:34 +00001928 reference_formats[i].magick,output_filename);
1929 reference_image=ReadImage(image_info,exception);
Cristye2879852017-08-13 12:53:45 -04001930 if ((reference_image == (Image *) NULL) ||
1931 (exception->severity >= ErrorException))
cristy3ed852e2009-09-05 21:47:34 +00001932 {
cristy1e604812011-05-19 18:07:50 +00001933 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
1934 GetMagickModule());
dirk775194c2014-05-17 19:47:54 +00001935 CatchException(exception);
cristy3ed852e2009-09-05 21:47:34 +00001936 (*fail)++;
1937 continue;
1938 }
1939 /*
1940 Write reference image.
1941 */
Cristy114baf52017-07-12 19:21:05 -04001942 (void) FormatLocaleString(reference_image->filename,MagickPathExtent,
1943 "%s:%s",reference_formats[i].magick,output_filename);
cristy3ed852e2009-09-05 21:47:34 +00001944 reference_image->depth=reference_types[j].depth;
cristy7998e5f2009-10-08 02:45:33 +00001945 reference_image->compression=reference_formats[i].compression;
cristy6f9e0d32011-08-28 16:32:09 +00001946 status=WriteImage(image_info,reference_image,exception);
dirk775194c2014-05-17 19:47:54 +00001947 if (status == MagickFalse ||exception->severity >= ErrorException)
cristy3ed852e2009-09-05 21:47:34 +00001948 {
cristy1e604812011-05-19 18:07:50 +00001949 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
1950 GetMagickModule());
dirk775194c2014-05-17 19:47:54 +00001951 CatchException(exception);
cristy3ed852e2009-09-05 21:47:34 +00001952 (*fail)++;
1953 reference_image=DestroyImage(reference_image);
1954 continue;
1955 }
1956 /*
1957 Read reconstruct image.
1958 */
cristy151b66d2015-04-15 10:50:31 +00001959 (void) FormatLocaleString(image_info->filename,MagickPathExtent,"%s:%s",
cristy3ed852e2009-09-05 21:47:34 +00001960 reference_formats[i].magick,output_filename);
1961 reconstruct_image=ReadImage(image_info,exception);
dirk775194c2014-05-17 19:47:54 +00001962 if (reconstruct_image == (Image *) NULL ||
Cristye2879852017-08-13 12:53:45 -04001963 (exception->severity >= ErrorException))
cristy3ed852e2009-09-05 21:47:34 +00001964 {
cristy1e604812011-05-19 18:07:50 +00001965 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
1966 GetMagickModule());
dirk775194c2014-05-17 19:47:54 +00001967 CatchException(exception);
cristy3ed852e2009-09-05 21:47:34 +00001968 (*fail)++;
1969 reference_image=DestroyImage(reference_image);
1970 continue;
1971 }
1972 /*
1973 Compare reference to reconstruct image.
1974 */
cristy93540ab2013-03-30 19:59:41 +00001975 fuzz=0.003; /* grayscale */
cristy3ed852e2009-09-05 21:47:34 +00001976 if (reference_formats[i].fuzz != 0.0)
1977 fuzz=reference_formats[i].fuzz;
cristy8a9106f2011-07-05 14:39:26 +00001978 difference_image=CompareImages(reference_image,reconstruct_image,
cristy864c6a82012-06-19 16:45:22 +00001979 RootMeanSquaredErrorMetric,&distortion,exception);
cristy3ed852e2009-09-05 21:47:34 +00001980 reconstruct_image=DestroyImage(reconstruct_image);
1981 reference_image=DestroyImage(reference_image);
dirk775194c2014-05-17 19:47:54 +00001982 if (difference_image == (Image *) NULL ||
Cristye2879852017-08-13 12:53:45 -04001983 (exception->severity >= ErrorException))
cristy3ed852e2009-09-05 21:47:34 +00001984 {
cristy1e604812011-05-19 18:07:50 +00001985 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
1986 GetMagickModule());
dirk775194c2014-05-17 19:47:54 +00001987 CatchException(exception);
cristy3ed852e2009-09-05 21:47:34 +00001988 (*fail)++;
1989 continue;
1990 }
1991 difference_image=DestroyImage(difference_image);
cristy14154222012-06-15 23:21:49 +00001992 if ((QuantumScale*distortion) > fuzz)
cristy3ed852e2009-09-05 21:47:34 +00001993 {
cristyb51dff52011-05-19 16:55:47 +00001994 (void) FormatLocaleFile(stdout,"... fail (with distortion %g).\n",
cristy14154222012-06-15 23:21:49 +00001995 QuantumScale*distortion);
cristy3ed852e2009-09-05 21:47:34 +00001996 (*fail)++;
1997 continue;
1998 }
cristyb51dff52011-05-19 16:55:47 +00001999 (void) FormatLocaleFile(stdout,"... pass.\n");
cristy3ed852e2009-09-05 21:47:34 +00002000 }
2001 }
cristyb51dff52011-05-19 16:55:47 +00002002 (void) FormatLocaleFile(stdout,
cristy75a26232010-06-03 17:15:02 +00002003 " summary: %.20g subtests; %.20g passed; %.20g failed.\n",(double) test,
2004 (double) (test-(*fail)),(double) *fail);
cristy3ed852e2009-09-05 21:47:34 +00002005 return(test);
2006}
2007
2008/*
2009%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2010% %
2011% %
2012% %
2013% V a l i d a t e I m p o r t E x p o r t P i x e l s %
2014% %
2015% %
2016% %
2017%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2018%
2019% ValidateImportExportPixels() validates the pixel import and export methods.
2020% It returns the number of validation tests that passed and failed.
2021%
2022% The format of the ValidateImportExportPixels method is:
2023%
cristybb503372010-05-27 20:51:26 +00002024% size_t ValidateImportExportPixels(ImageInfo *image_info,
cristy3ed852e2009-09-05 21:47:34 +00002025% const char *reference_filename,const char *output_filename,
cristybb503372010-05-27 20:51:26 +00002026% size_t *fail,ExceptionInfo *exception)
cristy3ed852e2009-09-05 21:47:34 +00002027%
2028% A description of each parameter follows:
2029%
2030% o image_info: the image info.
2031%
2032% o reference_filename: the reference image filename.
2033%
2034% o output_filename: the output image filename.
2035%
2036% o fail: return the number of validation tests that pass.
2037%
2038% o exception: return any errors or warnings in this structure.
2039%
2040*/
cristybb503372010-05-27 20:51:26 +00002041static size_t ValidateImportExportPixels(ImageInfo *image_info,
cristy0d772ed2012-11-10 14:34:50 +00002042 const char *reference_filename,const char *output_filename,size_t *fail,
2043 ExceptionInfo *exception)
cristy3ed852e2009-09-05 21:47:34 +00002044{
2045 double
2046 distortion;
2047
2048 Image
2049 *difference_image,
2050 *reference_image,
2051 *reconstruct_image;
2052
2053 MagickBooleanType
2054 status;
2055
cristybb503372010-05-27 20:51:26 +00002056 register ssize_t
cristy3ed852e2009-09-05 21:47:34 +00002057 i,
2058 j;
2059
2060 size_t
2061 length;
2062
2063 unsigned char
2064 *pixels;
2065
cristybb503372010-05-27 20:51:26 +00002066 size_t
cristy3ed852e2009-09-05 21:47:34 +00002067 test;
2068
2069 (void) output_filename;
2070 test=0;
cristy1e604812011-05-19 18:07:50 +00002071 (void) FormatLocaleFile(stdout,
2072 "validate the import and export of image pixels:\n");
cristy3ed852e2009-09-05 21:47:34 +00002073 for (i=0; reference_map[i] != (char *) NULL; i++)
2074 {
2075 for (j=0; reference_storage[j].type != UndefinedPixel; j++)
2076 {
2077 /*
2078 Generate reference image.
2079 */
2080 CatchException(exception);
cristyb51dff52011-05-19 16:55:47 +00002081 (void) FormatLocaleFile(stdout," test %.20g: %s/%s",(double) (test++),
cristy042ee782011-04-22 18:48:30 +00002082 reference_map[i],CommandOptionToMnemonic(MagickStorageOptions,
cristy3ed852e2009-09-05 21:47:34 +00002083 reference_storage[j].type));
2084 (void) CopyMagickString(image_info->filename,reference_filename,
cristy151b66d2015-04-15 10:50:31 +00002085 MagickPathExtent);
cristy3ed852e2009-09-05 21:47:34 +00002086 reference_image=ReadImage(image_info,exception);
Cristye2879852017-08-13 12:53:45 -04002087 if ((reference_image == (Image *) NULL) ||
2088 (exception->severity >= ErrorException))
cristy3ed852e2009-09-05 21:47:34 +00002089 {
cristy1e604812011-05-19 18:07:50 +00002090 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
2091 GetMagickModule());
dirk775194c2014-05-17 19:47:54 +00002092 CatchException(exception);
cristy3ed852e2009-09-05 21:47:34 +00002093 (*fail)++;
2094 continue;
2095 }
2096 if (LocaleNCompare(reference_map[i],"cmy",3) == 0)
cristy2c404222012-06-10 15:36:30 +00002097 (void) SetImageColorspace(reference_image,CMYKColorspace,exception);
cristy3ed852e2009-09-05 21:47:34 +00002098 length=strlen(reference_map[i])*reference_image->columns*
2099 reference_image->rows*reference_storage[j].quantum;
2100 pixels=(unsigned char *) AcquireQuantumMemory(length,sizeof(*pixels));
dirk775194c2014-05-17 19:47:54 +00002101 if (pixels == (unsigned char *) NULL ||
Cristye2879852017-08-13 12:53:45 -04002102 (exception->severity >= ErrorException))
cristy3ed852e2009-09-05 21:47:34 +00002103 {
cristy1e604812011-05-19 18:07:50 +00002104 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
2105 GetMagickModule());
dirk775194c2014-05-17 19:47:54 +00002106 CatchException(exception);
cristy3ed852e2009-09-05 21:47:34 +00002107 (*fail)++;
2108 reference_image=DestroyImage(reference_image);
2109 continue;
2110 }
2111 (void) ResetMagickMemory(pixels,0,length*sizeof(*pixels));
2112 status=ExportImagePixels(reference_image,0,0,reference_image->columns,
2113 reference_image->rows,reference_map[i],reference_storage[j].type,pixels,
2114 exception);
Cristye2879852017-08-13 12:53:45 -04002115 if (status == MagickFalse || (exception->severity >= ErrorException))
cristy3ed852e2009-09-05 21:47:34 +00002116 {
cristy1e604812011-05-19 18:07:50 +00002117 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
2118 GetMagickModule());
dirk775194c2014-05-17 19:47:54 +00002119 CatchException(exception);
cristy3ed852e2009-09-05 21:47:34 +00002120 (*fail)++;
2121 pixels=(unsigned char *) RelinquishMagickMemory(pixels);
2122 reference_image=DestroyImage(reference_image);
2123 continue;
2124 }
cristyd55c9352011-10-20 15:29:22 +00002125 (void) SetImageBackgroundColor(reference_image,exception);
cristy3ed852e2009-09-05 21:47:34 +00002126 status=ImportImagePixels(reference_image,0,0,reference_image->columns,
2127 reference_image->rows,reference_map[i],reference_storage[j].type,
cristy018f07f2011-09-04 21:15:19 +00002128 pixels,exception);
Cristye2879852017-08-13 12:53:45 -04002129 if (status == MagickFalse || (exception->severity >= ErrorException))
cristy3ed852e2009-09-05 21:47:34 +00002130 {
cristy1e604812011-05-19 18:07:50 +00002131 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
2132 GetMagickModule());
dirk775194c2014-05-17 19:47:54 +00002133 CatchException(exception);
cristy3ed852e2009-09-05 21:47:34 +00002134 (*fail)++;
2135 pixels=(unsigned char *) RelinquishMagickMemory(pixels);
2136 reference_image=DestroyImage(reference_image);
2137 continue;
2138 }
2139 /*
2140 Read reconstruct image.
2141 */
cristy9950d572011-10-01 18:22:35 +00002142 reconstruct_image=AcquireImage(image_info,exception);
cristy3ed852e2009-09-05 21:47:34 +00002143 (void) SetImageExtent(reconstruct_image,reference_image->columns,
cristy655a8d12011-08-05 19:24:28 +00002144 reference_image->rows,exception);
2145 (void) SetImageColorspace(reconstruct_image,reference_image->colorspace,
2146 exception);
cristyd55c9352011-10-20 15:29:22 +00002147 (void) SetImageBackgroundColor(reconstruct_image,exception);
cristy3ed852e2009-09-05 21:47:34 +00002148 status=ImportImagePixels(reconstruct_image,0,0,reconstruct_image->columns,
2149 reconstruct_image->rows,reference_map[i],reference_storage[j].type,
cristy018f07f2011-09-04 21:15:19 +00002150 pixels,exception);
cristy3ed852e2009-09-05 21:47:34 +00002151 pixels=(unsigned char *) RelinquishMagickMemory(pixels);
Cristye2879852017-08-13 12:53:45 -04002152 if (status == MagickFalse || (exception->severity >= ErrorException))
cristy3ed852e2009-09-05 21:47:34 +00002153 {
cristy1e604812011-05-19 18:07:50 +00002154 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
2155 GetMagickModule());
dirk775194c2014-05-17 19:47:54 +00002156 CatchException(exception);
cristy3ed852e2009-09-05 21:47:34 +00002157 (*fail)++;
2158 reference_image=DestroyImage(reference_image);
2159 continue;
2160 }
2161 /*
2162 Compare reference to reconstruct image.
2163 */
cristy8a9106f2011-07-05 14:39:26 +00002164 difference_image=CompareImages(reference_image,reconstruct_image,
cristy864c6a82012-06-19 16:45:22 +00002165 RootMeanSquaredErrorMetric,&distortion,exception);
cristy3ed852e2009-09-05 21:47:34 +00002166 reconstruct_image=DestroyImage(reconstruct_image);
2167 reference_image=DestroyImage(reference_image);
dirk775194c2014-05-17 19:47:54 +00002168 if (difference_image == (Image *) NULL ||
Cristye2879852017-08-13 12:53:45 -04002169 (exception->severity >= ErrorException))
cristy3ed852e2009-09-05 21:47:34 +00002170 {
cristy1e604812011-05-19 18:07:50 +00002171 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
2172 GetMagickModule());
dirk775194c2014-05-17 19:47:54 +00002173 CatchException(exception);
cristy3ed852e2009-09-05 21:47:34 +00002174 (*fail)++;
2175 continue;
2176 }
2177 difference_image=DestroyImage(difference_image);
cristy14154222012-06-15 23:21:49 +00002178 if ((QuantumScale*distortion) > 0.0)
cristy3ed852e2009-09-05 21:47:34 +00002179 {
cristyb51dff52011-05-19 16:55:47 +00002180 (void) FormatLocaleFile(stdout,"... fail (with distortion %g).\n",
cristy14154222012-06-15 23:21:49 +00002181 QuantumScale*distortion);
cristy3ed852e2009-09-05 21:47:34 +00002182 (*fail)++;
2183 continue;
2184 }
cristyb51dff52011-05-19 16:55:47 +00002185 (void) FormatLocaleFile(stdout,"... pass.\n");
cristy3ed852e2009-09-05 21:47:34 +00002186 }
2187 }
cristyb51dff52011-05-19 16:55:47 +00002188 (void) FormatLocaleFile(stdout,
cristy75a26232010-06-03 17:15:02 +00002189 " summary: %.20g subtests; %.20g passed; %.20g failed.\n",(double) test,
2190 (double) (test-(*fail)),(double) *fail);
cristy3ed852e2009-09-05 21:47:34 +00002191 return(test);
2192}
2193
2194/*
2195%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2196% %
2197% %
2198% %
2199% V a l i d a t e M o n t a g e C o m m a n d %
2200% %
2201% %
2202% %
2203%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2204%
2205% ValidateMontageCommand() validates the ImageMagick montage command line
2206% program and returns the number of validation tests that passed and failed.
2207%
2208% The format of the ValidateMontageCommand method is:
2209%
cristybb503372010-05-27 20:51:26 +00002210% size_t ValidateMontageCommand(ImageInfo *image_info,
cristy3ed852e2009-09-05 21:47:34 +00002211% const char *reference_filename,const char *output_filename,
cristybb503372010-05-27 20:51:26 +00002212% size_t *fail,ExceptionInfo *exception)
cristy3ed852e2009-09-05 21:47:34 +00002213%
2214% A description of each parameter follows:
2215%
2216% o image_info: the image info.
2217%
2218% o reference_filename: the reference image filename.
2219%
2220% o output_filename: the output image filename.
2221%
2222% o fail: return the number of validation tests that pass.
2223%
2224% o exception: return any errors or warnings in this structure.
2225%
2226*/
cristybb503372010-05-27 20:51:26 +00002227static size_t ValidateMontageCommand(ImageInfo *image_info,
cristy0d772ed2012-11-10 14:34:50 +00002228 const char *reference_filename,const char *output_filename,size_t *fail,
2229 ExceptionInfo *exception)
cristy3ed852e2009-09-05 21:47:34 +00002230{
2231 char
2232 **arguments,
cristy151b66d2015-04-15 10:50:31 +00002233 command[MagickPathExtent];
cristy3ed852e2009-09-05 21:47:34 +00002234
2235 int
2236 number_arguments;
2237
2238 MagickBooleanType
2239 status;
2240
cristybb503372010-05-27 20:51:26 +00002241 register ssize_t
cristy3ed852e2009-09-05 21:47:34 +00002242 i,
2243 j;
2244
cristybb503372010-05-27 20:51:26 +00002245 size_t
cristy3ed852e2009-09-05 21:47:34 +00002246 test;
2247
2248 test=0;
cristyb51dff52011-05-19 16:55:47 +00002249 (void) FormatLocaleFile(stdout,"validate montage command line program:\n");
cristy3ed852e2009-09-05 21:47:34 +00002250 for (i=0; montage_options[i] != (char *) NULL; i++)
2251 {
2252 CatchException(exception);
cristyb51dff52011-05-19 16:55:47 +00002253 (void) FormatLocaleFile(stdout," test %.20g: %s",(double) (test++),
cristy6aa47ad2010-05-28 19:32:32 +00002254 montage_options[i]);
cristy151b66d2015-04-15 10:50:31 +00002255 (void) FormatLocaleString(command,MagickPathExtent,"%s %s %s %s",
cristy3ed852e2009-09-05 21:47:34 +00002256 reference_filename,montage_options[i],reference_filename,
2257 output_filename);
2258 arguments=StringToArgv(command,&number_arguments);
2259 if (arguments == (char **) NULL)
2260 {
cristy1e604812011-05-19 18:07:50 +00002261 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
2262 GetMagickModule());
cristy3ed852e2009-09-05 21:47:34 +00002263 (*fail)++;
2264 continue;
2265 }
2266 status=MontageImageCommand(image_info,number_arguments,arguments,
2267 (char **) NULL,exception);
cristye42f6582012-02-11 17:59:50 +00002268 for (j=0; j < (ssize_t) number_arguments; j++)
cristy3ed852e2009-09-05 21:47:34 +00002269 arguments[j]=DestroyString(arguments[j]);
2270 arguments=(char **) RelinquishMagickMemory(arguments);
cristy35892192014-05-26 12:04:36 +00002271 if (status == MagickFalse)
cristy3ed852e2009-09-05 21:47:34 +00002272 {
cristy1e604812011-05-19 18:07:50 +00002273 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
2274 GetMagickModule());
cristy3ed852e2009-09-05 21:47:34 +00002275 (*fail)++;
2276 continue;
2277 }
cristyb51dff52011-05-19 16:55:47 +00002278 (void) FormatLocaleFile(stdout,"... pass.\n");
cristy3ed852e2009-09-05 21:47:34 +00002279 }
cristyb51dff52011-05-19 16:55:47 +00002280 (void) FormatLocaleFile(stdout,
cristy75a26232010-06-03 17:15:02 +00002281 " summary: %.20g subtests; %.20g passed; %.20g failed.\n",(double) test,
2282 (double) (test-(*fail)),(double) *fail);
cristy3ed852e2009-09-05 21:47:34 +00002283 return(test);
2284}
2285
2286/*
2287%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2288% %
2289% %
2290% %
2291% V a l i d a t e S t r e a m C o m m a n d %
2292% %
2293% %
2294% %
2295%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2296%
2297% ValidateStreamCommand() validates the ImageMagick stream command line
2298% program and returns the number of validation tests that passed and failed.
2299%
2300% The format of the ValidateStreamCommand method is:
2301%
cristybb503372010-05-27 20:51:26 +00002302% size_t ValidateStreamCommand(ImageInfo *image_info,
cristy3ed852e2009-09-05 21:47:34 +00002303% const char *reference_filename,const char *output_filename,
cristybb503372010-05-27 20:51:26 +00002304% size_t *fail,ExceptionInfo *exception)
cristy3ed852e2009-09-05 21:47:34 +00002305%
2306% A description of each parameter follows:
2307%
2308% o image_info: the image info.
2309%
2310% o reference_filename: the reference image filename.
2311%
2312% o output_filename: the output image filename.
2313%
2314% o fail: return the number of validation tests that pass.
2315%
2316% o exception: return any errors or warnings in this structure.
2317%
2318*/
cristybb503372010-05-27 20:51:26 +00002319static size_t ValidateStreamCommand(ImageInfo *image_info,
cristy0d772ed2012-11-10 14:34:50 +00002320 const char *reference_filename,const char *output_filename,size_t *fail,
2321 ExceptionInfo *exception)
cristy3ed852e2009-09-05 21:47:34 +00002322{
2323 char
2324 **arguments,
cristy151b66d2015-04-15 10:50:31 +00002325 command[MagickPathExtent];
cristy3ed852e2009-09-05 21:47:34 +00002326
2327 int
2328 number_arguments;
2329
2330 MagickBooleanType
2331 status;
2332
cristybb503372010-05-27 20:51:26 +00002333 register ssize_t
cristy3ed852e2009-09-05 21:47:34 +00002334 i,
2335 j;
2336
cristybb503372010-05-27 20:51:26 +00002337 size_t
cristy3ed852e2009-09-05 21:47:34 +00002338 test;
2339
2340 test=0;
cristyb51dff52011-05-19 16:55:47 +00002341 (void) FormatLocaleFile(stdout,"validate stream command line program:\n");
cristy3ed852e2009-09-05 21:47:34 +00002342 for (i=0; stream_options[i] != (char *) NULL; i++)
2343 {
2344 CatchException(exception);
cristyb51dff52011-05-19 16:55:47 +00002345 (void) FormatLocaleFile(stdout," test %.20g: %s",(double) (test++),
cristy6aa47ad2010-05-28 19:32:32 +00002346 stream_options[i]);
cristy151b66d2015-04-15 10:50:31 +00002347 (void) FormatLocaleString(command,MagickPathExtent,"%s %s %s",
cristy3ed852e2009-09-05 21:47:34 +00002348 stream_options[i],reference_filename,output_filename);
2349 arguments=StringToArgv(command,&number_arguments);
2350 if (arguments == (char **) NULL)
2351 {
cristy1e604812011-05-19 18:07:50 +00002352 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
2353 GetMagickModule());
cristy3ed852e2009-09-05 21:47:34 +00002354 (*fail)++;
2355 continue;
2356 }
2357 status=StreamImageCommand(image_info,number_arguments,arguments,
2358 (char **) NULL,exception);
cristye42f6582012-02-11 17:59:50 +00002359 for (j=0; j < (ssize_t) number_arguments; j++)
cristy3ed852e2009-09-05 21:47:34 +00002360 arguments[j]=DestroyString(arguments[j]);
2361 arguments=(char **) RelinquishMagickMemory(arguments);
cristy35892192014-05-26 12:04:36 +00002362 if (status == MagickFalse)
cristy3ed852e2009-09-05 21:47:34 +00002363 {
cristy1e604812011-05-19 18:07:50 +00002364 (void) FormatLocaleFile(stdout,"... fail @ %s/%s/%lu.\n",
2365 GetMagickModule());
cristy3ed852e2009-09-05 21:47:34 +00002366 (*fail)++;
2367 continue;
2368 }
cristyb51dff52011-05-19 16:55:47 +00002369 (void) FormatLocaleFile(stdout,"... pass.\n");
cristy3ed852e2009-09-05 21:47:34 +00002370 }
cristyb51dff52011-05-19 16:55:47 +00002371 (void) FormatLocaleFile(stdout,
cristy75a26232010-06-03 17:15:02 +00002372 " summary: %.20g subtests; %.20g passed; %.20g failed.\n",(double) test,
2373 (double) (test-(*fail)),(double) *fail);
cristy3ed852e2009-09-05 21:47:34 +00002374 return(test);
2375}
2376
2377/*
2378%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2379% %
2380% %
2381% %
2382% M a i n %
2383% %
2384% %
2385% %
2386%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2387%
2388%
2389*/
2390
2391static MagickBooleanType ValidateUsage(void)
2392{
2393 const char
2394 **p;
2395
2396 static const char
2397 *miscellaneous[]=
2398 {
2399 "-debug events display copious debugging information",
2400 "-help print program options",
2401 "-log format format of debugging information",
2402 "-validate type validation type",
2403 "-version print version information",
2404 (char *) NULL
2405 },
2406 *settings[]=
2407 {
2408 "-regard-warnings pay attention to warning messages",
2409 "-verbose print detailed information about the image",
2410 (char *) NULL
2411 };
2412
cristybb503372010-05-27 20:51:26 +00002413 (void) printf("Version: %s\n",GetMagickVersion((size_t *) NULL));
cristy3ed852e2009-09-05 21:47:34 +00002414 (void) printf("Copyright: %s\n\n",GetMagickCopyright());
cristyb28d6472009-10-17 20:13:35 +00002415 (void) printf("Features: %s\n",GetMagickFeatures());
cristy3ed852e2009-09-05 21:47:34 +00002416 (void) printf("Usage: %s [options ...] reference-file\n",GetClientName());
2417 (void) printf("\nValidate Settings:\n");
2418 for (p=settings; *p != (char *) NULL; p++)
2419 (void) printf(" %s\n",*p);
2420 (void) printf("\nMiscellaneous Options:\n");
2421 for (p=miscellaneous; *p != (char *) NULL; p++)
2422 (void) printf(" %s\n",*p);
2423 return(MagickTrue);
2424}
2425
2426int main(int argc,char **argv)
2427{
2428#define DestroyValidate() \
2429{ \
cristy3ed852e2009-09-05 21:47:34 +00002430 image_info=DestroyImageInfo(image_info); \
2431 exception=DestroyExceptionInfo(exception); \
2432}
2433#define ThrowValidateException(asperity,tag,option) \
2434{ \
2435 (void) ThrowMagickException(exception,GetMagickModule(),asperity,tag,"`%s'", \
2436 option); \
2437 CatchException(exception); \
2438 DestroyValidate(); \
2439 return(MagickFalse); \
2440}
2441
2442 char
cristy151b66d2015-04-15 10:50:31 +00002443 output_filename[MagickPathExtent],
2444 reference_filename[MagickPathExtent],
cristy3ed852e2009-09-05 21:47:34 +00002445 *option;
2446
2447 double
2448 elapsed_time,
2449 user_time;
2450
2451 ExceptionInfo
2452 *exception;
2453
2454 Image
2455 *reference_image;
2456
2457 ImageInfo
2458 *image_info;
2459
2460 MagickBooleanType
2461 regard_warnings,
2462 status;
2463
cristy9cafe592012-11-17 21:17:59 +00002464 MagickSizeType
2465 memory_resource,
2466 map_resource;
2467
cristybb503372010-05-27 20:51:26 +00002468 register ssize_t
cristy3ed852e2009-09-05 21:47:34 +00002469 i;
2470
2471 TimerInfo
2472 *timer;
2473
cristybb503372010-05-27 20:51:26 +00002474 size_t
cristy3ed852e2009-09-05 21:47:34 +00002475 fail,
2476 iterations,
2477 tests;
2478
2479 ValidateType
2480 type;
2481
2482 /*
2483 Validate the ImageMagick image processing suite.
2484 */
cristyb4d552c2012-12-13 15:42:17 +00002485 MagickCoreGenesis(*argv,MagickTrue);
cristyef2fdc62012-12-12 11:16:48 +00002486 (void) setlocale(LC_ALL,"");
2487 (void) setlocale(LC_NUMERIC,"C");
cristy3ed852e2009-09-05 21:47:34 +00002488 iterations=1;
2489 status=MagickFalse;
2490 type=AllValidate;
2491 regard_warnings=MagickFalse;
cristyd8862902011-02-20 21:12:27 +00002492 (void) regard_warnings;
cristy3ed852e2009-09-05 21:47:34 +00002493 exception=AcquireExceptionInfo();
2494 image_info=AcquireImageInfo();
cristy151b66d2015-04-15 10:50:31 +00002495 (void) CopyMagickString(image_info->filename,ReferenceFilename,MagickPathExtent);
cristybb503372010-05-27 20:51:26 +00002496 for (i=1; i < (ssize_t) argc; i++)
cristy3ed852e2009-09-05 21:47:34 +00002497 {
2498 option=argv[i];
cristy042ee782011-04-22 18:48:30 +00002499 if (IsCommandOption(option) == MagickFalse)
cristy3ed852e2009-09-05 21:47:34 +00002500 {
cristy151b66d2015-04-15 10:50:31 +00002501 (void) CopyMagickString(image_info->filename,option,MagickPathExtent);
cristy3ed852e2009-09-05 21:47:34 +00002502 continue;
2503 }
2504 switch (*(option+1))
2505 {
2506 case 'b':
2507 {
2508 if (LocaleCompare("bench",option+1) == 0)
2509 {
cristye27293e2009-12-18 02:53:20 +00002510 iterations=StringToUnsignedLong(argv[++i]);
cristy3ed852e2009-09-05 21:47:34 +00002511 break;
2512 }
2513 ThrowValidateException(OptionError,"UnrecognizedOption",option)
2514 }
2515 case 'd':
2516 {
2517 if (LocaleCompare("debug",option+1) == 0)
2518 {
2519 (void) SetLogEventMask(argv[++i]);
2520 break;
2521 }
2522 ThrowValidateException(OptionError,"UnrecognizedOption",option)
2523 }
2524 case 'h':
2525 {
2526 if (LocaleCompare("help",option+1) == 0)
2527 {
2528 (void) ValidateUsage();
2529 return(0);
2530 }
2531 ThrowValidateException(OptionError,"UnrecognizedOption",option)
2532 }
2533 case 'l':
2534 {
2535 if (LocaleCompare("log",option+1) == 0)
2536 {
2537 if (*option != '+')
2538 (void) SetLogFormat(argv[i+1]);
2539 break;
2540 }
2541 ThrowValidateException(OptionError,"UnrecognizedOption",option)
2542 }
2543 case 'r':
2544 {
2545 if (LocaleCompare("regard-warnings",option+1) == 0)
2546 {
2547 regard_warnings=MagickTrue;
2548 break;
2549 }
2550 ThrowValidateException(OptionError,"UnrecognizedOption",option)
2551 }
2552 case 'v':
2553 {
2554 if (LocaleCompare("validate",option+1) == 0)
2555 {
cristybb503372010-05-27 20:51:26 +00002556 ssize_t
cristy3ed852e2009-09-05 21:47:34 +00002557 validate;
2558
2559 if (*option == '+')
2560 break;
2561 i++;
cristye81f5522014-05-07 01:25:59 +00002562 if (i >= (ssize_t) argc)
cristy3ed852e2009-09-05 21:47:34 +00002563 ThrowValidateException(OptionError,"MissingArgument",option);
cristy042ee782011-04-22 18:48:30 +00002564 validate=ParseCommandOption(MagickValidateOptions,MagickFalse,
cristy3ed852e2009-09-05 21:47:34 +00002565 argv[i]);
2566 if (validate < 0)
2567 ThrowValidateException(OptionError,"UnrecognizedValidateType",
2568 argv[i]);
2569 type=(ValidateType) validate;
2570 break;
2571 }
2572 if ((LocaleCompare("version",option+1) == 0) ||
2573 (LocaleCompare("-version",option+1) == 0))
2574 {
cristyb51dff52011-05-19 16:55:47 +00002575 (void) FormatLocaleFile(stdout,"Version: %s\n",
cristybb503372010-05-27 20:51:26 +00002576 GetMagickVersion((size_t *) NULL));
cristy1e604812011-05-19 18:07:50 +00002577 (void) FormatLocaleFile(stdout,"Copyright: %s\n\n",
2578 GetMagickCopyright());
2579 (void) FormatLocaleFile(stdout,"Features: %s\n\n",
2580 GetMagickFeatures());
cristy3ed852e2009-09-05 21:47:34 +00002581 return(0);
2582 }
2583 ThrowValidateException(OptionError,"UnrecognizedOption",option)
2584 }
2585 default:
2586 ThrowValidateException(OptionError,"UnrecognizedOption",option)
2587 }
2588 }
cristy205e21f2009-09-28 19:01:46 +00002589 timer=(TimerInfo *) NULL;
cristy8b76e552009-09-28 18:20:20 +00002590 if (iterations > 1)
2591 timer=AcquireTimerInfo();
cristy3ed852e2009-09-05 21:47:34 +00002592 reference_image=ReadImage(image_info,exception);
2593 tests=0;
2594 fail=0;
2595 if (reference_image == (Image *) NULL)
2596 fail++;
2597 else
2598 {
2599 if (LocaleCompare(image_info->filename,ReferenceFilename) == 0)
2600 (void) CopyMagickString(reference_image->magick,ReferenceImageFormat,
cristy151b66d2015-04-15 10:50:31 +00002601 MagickPathExtent);
cristy3ed852e2009-09-05 21:47:34 +00002602 (void) AcquireUniqueFilename(reference_filename);
2603 (void) AcquireUniqueFilename(output_filename);
2604 (void) CopyMagickString(reference_image->filename,reference_filename,
cristy151b66d2015-04-15 10:50:31 +00002605 MagickPathExtent);
cristy6f9e0d32011-08-28 16:32:09 +00002606 status=WriteImage(image_info,reference_image,exception);
cristy3ed852e2009-09-05 21:47:34 +00002607 reference_image=DestroyImage(reference_image);
2608 if (status == MagickFalse)
2609 fail++;
2610 else
2611 {
cristyb51dff52011-05-19 16:55:47 +00002612 (void) FormatLocaleFile(stdout,"Version: %s\n",
cristybb503372010-05-27 20:51:26 +00002613 GetMagickVersion((size_t *) NULL));
cristyb51dff52011-05-19 16:55:47 +00002614 (void) FormatLocaleFile(stdout,"Copyright: %s\n\n",
cristy3ed852e2009-09-05 21:47:34 +00002615 GetMagickCopyright());
cristy1e604812011-05-19 18:07:50 +00002616 (void) FormatLocaleFile(stdout,
2617 "ImageMagick Validation Suite (%s)\n\n",CommandOptionToMnemonic(
2618 MagickValidateOptions,(ssize_t) type));
cristy80e870e2013-05-07 12:23:02 +00002619 if ((type & ColorspaceValidate) != 0)
cristy572e0972013-05-07 12:51:28 +00002620 tests+=ValidateColorspaces(image_info,&fail,exception);
cristy3ed852e2009-09-05 21:47:34 +00002621 if ((type & CompareValidate) != 0)
2622 tests+=ValidateCompareCommand(image_info,reference_filename,
2623 output_filename,&fail,exception);
2624 if ((type & CompositeValidate) != 0)
2625 tests+=ValidateCompositeCommand(image_info,reference_filename,
2626 output_filename,&fail,exception);
2627 if ((type & ConvertValidate) != 0)
2628 tests+=ValidateConvertCommand(image_info,reference_filename,
2629 output_filename,&fail,exception);
cristy247eebde2013-06-13 01:01:43 +00002630 if ((type & FormatsDiskValidate) != 0)
cristyf044a4d2012-11-17 21:10:20 +00002631 {
cristy9cafe592012-11-17 21:17:59 +00002632 memory_resource=SetMagickResourceLimit(MemoryResource,0);
cristy247eebde2013-06-13 01:01:43 +00002633 map_resource=SetMagickResourceLimit(MapResource,0);
2634 (void) FormatLocaleFile(stdout,"[pixel-cache: disk] ");
cristyf044a4d2012-11-17 21:10:20 +00002635 tests+=ValidateImageFormatsInMemory(image_info,reference_filename,
2636 output_filename,&fail,exception);
2637 (void) FormatLocaleFile(stdout,"[pixel-cache: disk] ");
cristy247eebde2013-06-13 01:01:43 +00002638 tests+=ValidateImageFormatsOnDisk(image_info,reference_filename,
cristyf044a4d2012-11-17 21:10:20 +00002639 output_filename,&fail,exception);
cristy9cafe592012-11-17 21:17:59 +00002640 (void) SetMagickResourceLimit(MemoryResource,memory_resource);
2641 (void) SetMagickResourceLimit(MapResource,map_resource);
cristyf044a4d2012-11-17 21:10:20 +00002642 }
cristy247eebde2013-06-13 01:01:43 +00002643 if ((type & FormatsMapValidate) != 0)
cristybda201a2012-11-08 16:44:00 +00002644 {
cristy247eebde2013-06-13 01:01:43 +00002645 memory_resource=SetMagickResourceLimit(MemoryResource,0);
2646 (void) FormatLocaleFile(stdout,"[pixel-cache: memory-mapped] ");
2647 tests+=ValidateImageFormatsInMemory(image_info,reference_filename,
cristybda201a2012-11-08 16:44:00 +00002648 output_filename,&fail,exception);
cristy34c290f2012-11-09 00:44:12 +00002649 (void) FormatLocaleFile(stdout,"[pixel-cache: memory-mapped] ");
cristybda201a2012-11-08 16:44:00 +00002650 tests+=ValidateImageFormatsOnDisk(image_info,reference_filename,
2651 output_filename,&fail,exception);
cristy9cafe592012-11-17 21:17:59 +00002652 (void) SetMagickResourceLimit(MemoryResource,memory_resource);
cristy247eebde2013-06-13 01:01:43 +00002653 }
2654 if ((type & FormatsMemoryValidate) != 0)
2655 {
2656 (void) FormatLocaleFile(stdout,"[pixel-cache: memory] ");
2657 tests+=ValidateImageFormatsInMemory(image_info,reference_filename,
2658 output_filename,&fail,exception);
2659 (void) FormatLocaleFile(stdout,"[pixel-cache: memory] ");
2660 tests+=ValidateImageFormatsOnDisk(image_info,reference_filename,
2661 output_filename,&fail,exception);
cristybda201a2012-11-08 16:44:00 +00002662 }
cristy3ed852e2009-09-05 21:47:34 +00002663 if ((type & IdentifyValidate) != 0)
2664 tests+=ValidateIdentifyCommand(image_info,reference_filename,
2665 output_filename,&fail,exception);
2666 if ((type & ImportExportValidate) != 0)
2667 tests+=ValidateImportExportPixels(image_info,reference_filename,
2668 output_filename,&fail,exception);
2669 if ((type & MontageValidate) != 0)
2670 tests+=ValidateMontageCommand(image_info,reference_filename,
2671 output_filename,&fail,exception);
2672 if ((type & StreamValidate) != 0)
2673 tests+=ValidateStreamCommand(image_info,reference_filename,
2674 output_filename,&fail,exception);
cristyb51dff52011-05-19 16:55:47 +00002675 (void) FormatLocaleFile(stdout,
cristy75a26232010-06-03 17:15:02 +00002676 "validation suite: %.20g tests; %.20g passed; %.20g failed.\n",
2677 (double) tests,(double) (tests-fail),(double) fail);
cristy3ed852e2009-09-05 21:47:34 +00002678 }
2679 (void) RelinquishUniqueFileResource(output_filename);
2680 (void) RelinquishUniqueFileResource(reference_filename);
2681 }
2682 if (exception->severity != UndefinedException)
2683 CatchException(exception);
2684 if (iterations > 1)
2685 {
2686 elapsed_time=GetElapsedTime(timer);
2687 user_time=GetUserTime(timer);
cristyb51dff52011-05-19 16:55:47 +00002688 (void) FormatLocaleFile(stderr,
cristy75a26232010-06-03 17:15:02 +00002689 "Performance: %.20gi %gips %0.3fu %ld:%02ld.%03ld\n",(double)
cristy6aa47ad2010-05-28 19:32:32 +00002690 iterations,1.0*iterations/elapsed_time,user_time,(long)
2691 (elapsed_time/60.0),(long) ceil(fmod(elapsed_time,60.0)),
2692 (long) (1000.0*(elapsed_time-floor(elapsed_time))));
cristy8b76e552009-09-28 18:20:20 +00002693 timer=DestroyTimerInfo(timer);
cristy3ed852e2009-09-05 21:47:34 +00002694 }
2695 DestroyValidate();
2696 MagickCoreTerminus();
2697 return(fail == 0 ? 0 : 1);
2698}