]> git.ipfire.org Git - thirdparty/cups.git/blob - cups/testppd.c
Merge changes from CUPS 1.5svn-r9105.
[thirdparty/cups.git] / cups / testppd.c
1 /*
2 * "$Id: testppd.c 7897 2008-09-02 19:33:19Z mike $"
3 *
4 * PPD test program for CUPS.
5 *
6 * Copyright 2007-2010 by Apple Inc.
7 * Copyright 1997-2006 by Easy Software Products.
8 *
9 * These coded instructions, statements, and computer programs are the
10 * property of Apple Inc. and are protected by Federal copyright
11 * law. Distribution and use rights are outlined in the file "LICENSE.txt"
12 * which should have been included with this file. If this file is
13 * file is missing or damaged, see the license at "http://www.cups.org/".
14 *
15 * This file is subject to the Apple OS-Developed Software exception.
16 *
17 * Contents:
18 *
19 * main() - Main entry.
20 */
21
22 /*
23 * Include necessary headers...
24 */
25
26 #include "cups-private.h"
27 #include <sys/stat.h>
28 #ifdef WIN32
29 # include <io.h>
30 #else
31 # include <unistd.h>
32 # include <fcntl.h>
33 #endif /* WIN32 */
34
35
36 /*
37 * Test data...
38 */
39
40 static const char *default_code =
41 "[{\n"
42 "%%BeginFeature: *InstalledDuplexer False\n"
43 "%%EndFeature\n"
44 "} stopped cleartomark\n"
45 "[{\n"
46 "%%BeginFeature: *PageRegion Letter\n"
47 "PageRegion=Letter\n"
48 "%%EndFeature\n"
49 "} stopped cleartomark\n"
50 "[{\n"
51 "%%BeginFeature: *InputSlot Tray\n"
52 "InputSlot=Tray\n"
53 "%%EndFeature\n"
54 "} stopped cleartomark\n"
55 "[{\n"
56 "%%BeginFeature: *MediaType Plain\n"
57 "MediaType=Plain\n"
58 "%%EndFeature\n"
59 "} stopped cleartomark\n"
60 "[{\n"
61 "%%BeginFeature: *IntOption None\n"
62 "%%EndFeature\n"
63 "} stopped cleartomark\n"
64 "[{\n"
65 "%%BeginFeature: *StringOption None\n"
66 "%%EndFeature\n"
67 "} stopped cleartomark\n";
68
69 static const char *custom_code =
70 "[{\n"
71 "%%BeginFeature: *InstalledDuplexer False\n"
72 "%%EndFeature\n"
73 "} stopped cleartomark\n"
74 "[{\n"
75 "%%BeginFeature: *InputSlot Tray\n"
76 "InputSlot=Tray\n"
77 "%%EndFeature\n"
78 "} stopped cleartomark\n"
79 "[{\n"
80 "%%BeginFeature: *MediaType Plain\n"
81 "MediaType=Plain\n"
82 "%%EndFeature\n"
83 "} stopped cleartomark\n"
84 "[{\n"
85 "%%BeginFeature: *IntOption None\n"
86 "%%EndFeature\n"
87 "} stopped cleartomark\n"
88 "[{\n"
89 "%%BeginFeature: *CustomStringOption True\n"
90 "(value\\0502\\051)\n"
91 "(value 1)\n"
92 "StringOption=Custom\n"
93 "%%EndFeature\n"
94 "} stopped cleartomark\n"
95 "[{\n"
96 "%%BeginFeature: *CustomPageSize True\n"
97 "400\n"
98 "500\n"
99 "0\n"
100 "0\n"
101 "0\n"
102 "PageSize=Custom\n"
103 "%%EndFeature\n"
104 "} stopped cleartomark\n";
105
106 static const char *default2_code =
107 "[{\n"
108 "%%BeginFeature: *InstalledDuplexer False\n"
109 "%%EndFeature\n"
110 "} stopped cleartomark\n"
111 "[{\n"
112 "%%BeginFeature: *InputSlot Tray\n"
113 "InputSlot=Tray\n"
114 "%%EndFeature\n"
115 "} stopped cleartomark\n"
116 "[{\n"
117 "%%BeginFeature: *Quality Normal\n"
118 "Quality=Normal\n"
119 "%%EndFeature\n"
120 "} stopped cleartomark\n"
121 "[{\n"
122 "%%BeginFeature: *IntOption None\n"
123 "%%EndFeature\n"
124 "} stopped cleartomark\n"
125 "[{\n"
126 "%%BeginFeature: *StringOption None\n"
127 "%%EndFeature\n"
128 "} stopped cleartomark\n";
129
130
131 /*
132 * 'main()' - Main entry.
133 */
134
135 int /* O - Exit status */
136 main(int argc, /* I - Number of command-line arguments */
137 char *argv[]) /* I - Command-line arguments */
138 {
139 int i; /* Looping var */
140 ppd_file_t *ppd; /* PPD file loaded from disk */
141 int status; /* Status of tests (0 = success, 1 = fail) */
142 int conflicts; /* Number of conflicts */
143 char *s; /* String */
144 char buffer[8192]; /* String buffer */
145 const char *text, /* Localized text */
146 *val; /* Option value */
147 int num_options; /* Number of options */
148 cups_option_t *options; /* Options */
149 ppd_size_t minsize, /* Minimum size */
150 maxsize, /* Maximum size */
151 *size; /* Current size */
152 ppd_attr_t *attr; /* Current attribute */
153
154
155 status = 0;
156
157 if (argc == 1)
158 {
159 /*
160 * Setup directories for locale stuff...
161 */
162
163 if (access("locale", 0))
164 {
165 mkdir("locale", 0777);
166 mkdir("locale/fr", 0777);
167 symlink("../../../locale/cups_fr.po", "locale/fr/cups_fr.po");
168 mkdir("locale/zh_TW", 0777);
169 symlink("../../../locale/cups_zh_TW.po", "locale/zh_TW/cups_zh_TW.po");
170 }
171
172 putenv("LOCALEDIR=locale");
173 putenv("SOFTWARE=CUPS");
174
175 /*
176 * Do tests with test.ppd...
177 */
178
179 fputs("ppdOpenFile(test.ppd): ", stdout);
180
181 if ((ppd = ppdOpenFile("test.ppd")) != NULL)
182 puts("PASS");
183 else
184 {
185 ppd_status_t err; /* Last error in file */
186 int line; /* Line number in file */
187
188
189 status ++;
190 err = ppdLastError(&line);
191
192 printf("FAIL (%s on line %d)\n", ppdErrorString(err), line);
193 }
194
195 fputs("ppdFindAttr(wildcard): ", stdout);
196 if ((attr = ppdFindAttr(ppd, "cupsTest", NULL)) == NULL)
197 {
198 status ++;
199 puts("FAIL (not found)");
200 }
201 else if (strcmp(attr->name, "cupsTest") || strcmp(attr->spec, "Foo"))
202 {
203 status ++;
204 printf("FAIL (got \"%s %s\")\n", attr->name, attr->spec);
205 }
206 else
207 puts("PASS");
208
209 fputs("ppdFindNextAttr(wildcard): ", stdout);
210 if ((attr = ppdFindNextAttr(ppd, "cupsTest", NULL)) == NULL)
211 {
212 status ++;
213 puts("FAIL (not found)");
214 }
215 else if (strcmp(attr->name, "cupsTest") || strcmp(attr->spec, "Bar"))
216 {
217 status ++;
218 printf("FAIL (got \"%s %s\")\n", attr->name, attr->spec);
219 }
220 else
221 puts("PASS");
222
223 fputs("ppdFindAttr(Foo): ", stdout);
224 if ((attr = ppdFindAttr(ppd, "cupsTest", "Foo")) == NULL)
225 {
226 status ++;
227 puts("FAIL (not found)");
228 }
229 else if (strcmp(attr->name, "cupsTest") || strcmp(attr->spec, "Foo"))
230 {
231 status ++;
232 printf("FAIL (got \"%s %s\")\n", attr->name, attr->spec);
233 }
234 else
235 puts("PASS");
236
237 fputs("ppdFindNextAttr(Foo): ", stdout);
238 if ((attr = ppdFindNextAttr(ppd, "cupsTest", "Foo")) != NULL)
239 {
240 status ++;
241 printf("FAIL (got \"%s %s\")\n", attr->name, attr->spec);
242 }
243 else
244 puts("PASS");
245
246 fputs("ppdMarkDefaults: ", stdout);
247 ppdMarkDefaults(ppd);
248
249 if ((conflicts = ppdConflicts(ppd)) == 0)
250 puts("PASS");
251 else
252 {
253 status ++;
254 printf("FAIL (%d conflicts)\n", conflicts);
255 }
256
257 fputs("ppdEmitString (defaults): ", stdout);
258 if ((s = ppdEmitString(ppd, PPD_ORDER_ANY, 0.0)) != NULL &&
259 !strcmp(s, default_code))
260 puts("PASS");
261 else
262 {
263 status ++;
264 printf("FAIL (%d bytes instead of %d)\n", s ? (int)strlen(s) : 0,
265 (int)strlen(default_code));
266
267 if (s)
268 puts(s);
269 }
270
271 if (s)
272 free(s);
273
274 fputs("ppdEmitString (custom size and string): ", stdout);
275 ppdMarkOption(ppd, "PageSize", "Custom.400x500");
276 ppdMarkOption(ppd, "StringOption", "{String1=\"value 1\" String2=value(2)}");
277
278 if ((s = ppdEmitString(ppd, PPD_ORDER_ANY, 0.0)) != NULL &&
279 !strcmp(s, custom_code))
280 puts("PASS");
281 else
282 {
283 status ++;
284 printf("FAIL (%d bytes instead of %d)\n", s ? (int)strlen(s) : 0,
285 (int)strlen(custom_code));
286
287 if (s)
288 puts(s);
289 }
290
291 if (s)
292 free(s);
293
294 /*
295 * Test constraints...
296 */
297
298 fputs("cupsGetConflicts(InputSlot=Envelope): ", stdout);
299 ppdMarkOption(ppd, "PageSize", "Letter");
300
301 num_options = cupsGetConflicts(ppd, "InputSlot", "Envelope", &options);
302 if (num_options != 2 ||
303 (val = cupsGetOption("PageRegion", num_options, options)) == NULL ||
304 strcasecmp(val, "Letter") ||
305 (val = cupsGetOption("PageSize", num_options, options)) == NULL ||
306 strcasecmp(val, "Letter"))
307 {
308 printf("FAIL (%d options:", num_options);
309 for (i = 0; i < num_options; i ++)
310 printf(" %s=%s", options[i].name, options[i].value);
311 puts(")");
312 status ++;
313 }
314 else
315 puts("PASS");
316
317 fputs("ppdConflicts(): ", stdout);
318 ppdMarkOption(ppd, "InputSlot", "Envelope");
319
320 if ((conflicts = ppdConflicts(ppd)) == 2)
321 puts("PASS (2)");
322 else
323 {
324 printf("FAIL (%d)\n", conflicts);
325 status ++;
326 }
327
328 fputs("cupsResolveConflicts(InputSlot=Envelope): ", stdout);
329 num_options = 0;
330 options = NULL;
331 if (!cupsResolveConflicts(ppd, "InputSlot", "Envelope", &num_options,
332 &options))
333 {
334 puts("FAIL (Unable to resolve)");
335 status ++;
336 }
337 else if (num_options != 2 ||
338 !cupsGetOption("PageSize", num_options, options))
339 {
340 printf("FAIL (%d options:", num_options);
341 for (i = 0; i < num_options; i ++)
342 printf(" %s=%s", options[i].name, options[i].value);
343 puts(")");
344 status ++;
345 }
346 else
347 puts("PASS (Resolved by changing PageSize)");
348
349 cupsFreeOptions(num_options, options);
350
351 fputs("cupsResolveConflicts(No option/choice): ", stdout);
352 num_options = 0;
353 options = NULL;
354 if (cupsResolveConflicts(ppd, NULL, NULL, &num_options, &options) &&
355 num_options == 1 && !strcasecmp(options[0].name, "InputSlot") &&
356 !strcasecmp(options[0].value, "Tray"))
357 puts("PASS (Resolved by changing InputSlot)");
358 else if (num_options > 0)
359 {
360 printf("FAIL (%d options:", num_options);
361 for (i = 0; i < num_options; i ++)
362 printf(" %s=%s", options[i].name, options[i].value);
363 puts(")");
364 status ++;
365 }
366 else
367 {
368 puts("FAIL (Unable to resolve)");
369 status ++;
370 }
371 cupsFreeOptions(num_options, options);
372
373 fputs("ppdInstallableConflict(): ", stdout);
374 if (ppdInstallableConflict(ppd, "Duplex", "DuplexNoTumble") &&
375 !ppdInstallableConflict(ppd, "Duplex", "None"))
376 puts("PASS");
377 else if (!ppdInstallableConflict(ppd, "Duplex", "DuplexNoTumble"))
378 {
379 puts("FAIL (Duplex=DuplexNoTumble did not conflict)");
380 status ++;
381 }
382 else
383 {
384 puts("FAIL (Duplex=None conflicted)");
385 status ++;
386 }
387
388 /*
389 * ppdPageSizeLimits
390 */
391
392 fputs("ppdPageSizeLimits: ", stdout);
393 if (ppdPageSizeLimits(ppd, &minsize, &maxsize))
394 {
395 if (minsize.width != 36 || minsize.length != 36 ||
396 maxsize.width != 1080 || maxsize.length != 86400)
397 {
398 printf("FAIL (got min=%.0fx%.0f, max=%.0fx%.0f, "
399 "expected min=36x36, max=1080x86400)\n", minsize.width,
400 minsize.length, maxsize.width, maxsize.length);
401 status ++;
402 }
403 else
404 puts("PASS");
405 }
406 else
407 {
408 puts("FAIL (returned 0)");
409 status ++;
410 }
411
412 /*
413 * cupsMarkOptions with PWG and IPP size names.
414 */
415
416 fputs("cupsMarkOptions(media=iso-a4): ", stdout);
417 num_options = cupsAddOption("media", "iso-a4", 0, &options);
418 cupsMarkOptions(ppd, num_options, options);
419 cupsFreeOptions(num_options, options);
420
421 size = ppdPageSize(ppd, NULL);
422 if (!size || strcmp(size->name, "A4"))
423 {
424 printf("FAIL (%s)\n", size ? size->name : "unknown");
425 status ++;
426 }
427 else
428 puts("PASS");
429
430 fputs("cupsMarkOptions(media=na_letter_8.5x11in): ", stdout);
431 num_options = cupsAddOption("media", "na_letter_8.5x11in", 0, &options);
432 cupsMarkOptions(ppd, num_options, options);
433 cupsFreeOptions(num_options, options);
434
435 size = ppdPageSize(ppd, NULL);
436 if (!size || strcmp(size->name, "Letter"))
437 {
438 printf("FAIL (%s)\n", size ? size->name : "unknown");
439 status ++;
440 }
441 else
442 puts("PASS");
443
444 /*
445 * Test localization...
446 */
447
448 fputs("ppdLocalizeIPPReason(text): ", stdout);
449 if (ppdLocalizeIPPReason(ppd, "foo", NULL, buffer, sizeof(buffer)) &&
450 !strcmp(buffer, "Foo Reason"))
451 puts("PASS");
452 else
453 {
454 status ++;
455 printf("FAIL (\"%s\" instead of \"Foo Reason\")\n", buffer);
456 }
457
458 fputs("ppdLocalizeIPPReason(http): ", stdout);
459 if (ppdLocalizeIPPReason(ppd, "foo", "http", buffer, sizeof(buffer)) &&
460 !strcmp(buffer, "http://foo/bar.html"))
461 puts("PASS");
462 else
463 {
464 status ++;
465 printf("FAIL (\"%s\" instead of \"http://foo/bar.html\")\n", buffer);
466 }
467
468 fputs("ppdLocalizeIPPReason(help): ", stdout);
469 if (ppdLocalizeIPPReason(ppd, "foo", "help", buffer, sizeof(buffer)) &&
470 !strcmp(buffer, "help:anchor='foo'%20bookID=Vendor%20Help"))
471 puts("PASS");
472 else
473 {
474 status ++;
475 printf("FAIL (\"%s\" instead of \"help:anchor='foo'%%20bookID=Vendor%%20Help\")\n", buffer);
476 }
477
478 fputs("ppdLocalizeIPPReason(file): ", stdout);
479 if (ppdLocalizeIPPReason(ppd, "foo", "file", buffer, sizeof(buffer)) &&
480 !strcmp(buffer, "/help/foo/bar.html"))
481 puts("PASS");
482 else
483 {
484 status ++;
485 printf("FAIL (\"%s\" instead of \"/help/foo/bar.html\")\n", buffer);
486 }
487
488 putenv("LANG=fr");
489 putenv("LC_ALL=fr");
490 putenv("LC_CTYPE=fr");
491 putenv("LC_MESSAGES=fr");
492
493 fputs("ppdLocalizeIPPReason(fr text): ", stdout);
494 if (ppdLocalizeIPPReason(ppd, "foo", NULL, buffer, sizeof(buffer)) &&
495 !strcmp(buffer, "La Long Foo Reason"))
496 puts("PASS");
497 else
498 {
499 status ++;
500 printf("FAIL (\"%s\" instead of \"La Long Foo Reason\")\n", buffer);
501 }
502
503 putenv("LANG=zh_TW");
504 putenv("LC_ALL=zh_TW");
505 putenv("LC_CTYPE=zh_TW");
506 putenv("LC_MESSAGES=zh_TW");
507
508 fputs("ppdLocalizeIPPReason(zh_TW text): ", stdout);
509 if (ppdLocalizeIPPReason(ppd, "foo", NULL, buffer, sizeof(buffer)) &&
510 !strcmp(buffer, "Number 1 Foo Reason"))
511 puts("PASS");
512 else
513 {
514 status ++;
515 printf("FAIL (\"%s\" instead of \"Number 1 Foo Reason\")\n", buffer);
516 }
517
518 /*
519 * cupsMarkerName localization...
520 */
521
522 putenv("LANG=en");
523 putenv("LC_ALL=en");
524 putenv("LC_CTYPE=en");
525 putenv("LC_MESSAGES=en");
526
527 fputs("ppdLocalizeMarkerName(bogus): ", stdout);
528
529 if ((text = ppdLocalizeMarkerName(ppd, "bogus")) != NULL)
530 {
531 status ++;
532 printf("FAIL (\"%s\" instead of NULL)\n", text);
533 }
534 else
535 puts("PASS");
536
537 fputs("ppdLocalizeMarkerName(cyan): ", stdout);
538
539 if ((text = ppdLocalizeMarkerName(ppd, "cyan")) != NULL &&
540 !strcmp(text, "Cyan Toner"))
541 puts("PASS");
542 else
543 {
544 status ++;
545 printf("FAIL (\"%s\" instead of \"Cyan Toner\")\n",
546 text ? text : "(null)");
547 }
548
549 putenv("LANG=fr");
550 putenv("LC_ALL=fr");
551 putenv("LC_CTYPE=fr");
552 putenv("LC_MESSAGES=fr");
553
554 fputs("ppdLocalizeMarkerName(fr cyan): ", stdout);
555 if ((text = ppdLocalizeMarkerName(ppd, "cyan")) != NULL &&
556 !strcmp(text, "La Toner Cyan"))
557 puts("PASS");
558 else
559 {
560 status ++;
561 printf("FAIL (\"%s\" instead of \"La Toner Cyan\")\n",
562 text ? text : "(null)");
563 }
564
565 putenv("LANG=zh_TW");
566 putenv("LC_ALL=zh_TW");
567 putenv("LC_CTYPE=zh_TW");
568 putenv("LC_MESSAGES=zh_TW");
569
570 fputs("ppdLocalizeMarkerName(zh_TW cyan): ", stdout);
571 if ((text = ppdLocalizeMarkerName(ppd, "cyan")) != NULL &&
572 !strcmp(text, "Number 1 Cyan Toner"))
573 puts("PASS");
574 else
575 {
576 status ++;
577 printf("FAIL (\"%s\" instead of \"Number 1 Cyan Toner\")\n",
578 text ? text : "(null)");
579 }
580
581 ppdClose(ppd);
582
583 /*
584 * Test new constraints...
585 */
586
587 fputs("ppdOpenFile(test2.ppd): ", stdout);
588
589 if ((ppd = ppdOpenFile("test2.ppd")) != NULL)
590 puts("PASS");
591 else
592 {
593 ppd_status_t err; /* Last error in file */
594 int line; /* Line number in file */
595
596
597 status ++;
598 err = ppdLastError(&line);
599
600 printf("FAIL (%s on line %d)\n", ppdErrorString(err), line);
601 }
602
603 fputs("ppdMarkDefaults: ", stdout);
604 ppdMarkDefaults(ppd);
605
606 if ((conflicts = ppdConflicts(ppd)) == 0)
607 puts("PASS");
608 else
609 {
610 status ++;
611 printf("FAIL (%d conflicts)\n", conflicts);
612 }
613
614 fputs("ppdEmitString (defaults): ", stdout);
615 if ((s = ppdEmitString(ppd, PPD_ORDER_ANY, 0.0)) != NULL &&
616 !strcmp(s, default2_code))
617 puts("PASS");
618 else
619 {
620 status ++;
621 printf("FAIL (%d bytes instead of %d)\n", s ? (int)strlen(s) : 0,
622 (int)strlen(default2_code));
623
624 if (s)
625 puts(s);
626 }
627
628 if (s)
629 free(s);
630
631 fputs("ppdConflicts(): ", stdout);
632 ppdMarkOption(ppd, "PageSize", "Env10");
633 ppdMarkOption(ppd, "InputSlot", "Envelope");
634 ppdMarkOption(ppd, "Quality", "Photo");
635
636 if ((conflicts = ppdConflicts(ppd)) == 1)
637 puts("PASS (1)");
638 else
639 {
640 printf("FAIL (%d)\n", conflicts);
641 status ++;
642 }
643
644 fputs("cupsResolveConflicts(Quality=Photo): ", stdout);
645 num_options = 0;
646 options = NULL;
647 if (cupsResolveConflicts(ppd, "Quality", "Photo", &num_options,
648 &options))
649 {
650 printf("FAIL (%d options:", num_options);
651 for (i = 0; i < num_options; i ++)
652 printf(" %s=%s", options[i].name, options[i].value);
653 puts(")");
654 status ++;
655 }
656 else
657 puts("PASS (Unable to resolve)");
658 cupsFreeOptions(num_options, options);
659
660 fputs("cupsResolveConflicts(No option/choice): ", stdout);
661 num_options = 0;
662 options = NULL;
663 if (cupsResolveConflicts(ppd, NULL, NULL, &num_options, &options) &&
664 num_options == 1 && !strcasecmp(options->name, "Quality") &&
665 !strcasecmp(options->value, "Normal"))
666 puts("PASS");
667 else if (num_options > 0)
668 {
669 printf("FAIL (%d options:", num_options);
670 for (i = 0; i < num_options; i ++)
671 printf(" %s=%s", options[i].name, options[i].value);
672 puts(")");
673 status ++;
674 }
675 else
676 {
677 puts("FAIL (Unable to resolve!)");
678 status ++;
679 }
680 cupsFreeOptions(num_options, options);
681
682 fputs("cupsResolveConflicts(loop test): ", stdout);
683 ppdMarkOption(ppd, "PageSize", "A4");
684 ppdMarkOption(ppd, "InputSlot", "Tray");
685 ppdMarkOption(ppd, "Quality", "Photo");
686 num_options = 0;
687 options = NULL;
688 if (!cupsResolveConflicts(ppd, NULL, NULL, &num_options, &options))
689 puts("PASS");
690 else if (num_options > 0)
691 {
692 printf("FAIL (%d options:", num_options);
693 for (i = 0; i < num_options; i ++)
694 printf(" %s=%s", options[i].name, options[i].value);
695 puts(")");
696 }
697 else
698 puts("FAIL (No conflicts!)");
699
700 fputs("ppdInstallableConflict(): ", stdout);
701 if (ppdInstallableConflict(ppd, "Duplex", "DuplexNoTumble") &&
702 !ppdInstallableConflict(ppd, "Duplex", "None"))
703 puts("PASS");
704 else if (!ppdInstallableConflict(ppd, "Duplex", "DuplexNoTumble"))
705 {
706 puts("FAIL (Duplex=DuplexNoTumble did not conflict)");
707 status ++;
708 }
709 else
710 {
711 puts("FAIL (Duplex=None conflicted)");
712 status ++;
713 }
714
715 /*
716 * ppdPageSizeLimits
717 */
718
719 ppdMarkDefaults(ppd);
720
721 fputs("ppdPageSizeLimits(default): ", stdout);
722 if (ppdPageSizeLimits(ppd, &minsize, &maxsize))
723 {
724 if (minsize.width != 36 || minsize.length != 36 ||
725 maxsize.width != 1080 || maxsize.length != 86400)
726 {
727 printf("FAIL (got min=%.0fx%.0f, max=%.0fx%.0f, "
728 "expected min=36x36, max=1080x86400)\n", minsize.width,
729 minsize.length, maxsize.width, maxsize.length);
730 status ++;
731 }
732 else
733 puts("PASS");
734 }
735 else
736 {
737 puts("FAIL (returned 0)");
738 status ++;
739 }
740
741 ppdMarkOption(ppd, "InputSlot", "Manual");
742
743 fputs("ppdPageSizeLimits(InputSlot=Manual): ", stdout);
744 if (ppdPageSizeLimits(ppd, &minsize, &maxsize))
745 {
746 if (minsize.width != 100 || minsize.length != 100 ||
747 maxsize.width != 1000 || maxsize.length != 1000)
748 {
749 printf("FAIL (got min=%.0fx%.0f, max=%.0fx%.0f, "
750 "expected min=100x100, max=1000x1000)\n", minsize.width,
751 minsize.length, maxsize.width, maxsize.length);
752 status ++;
753 }
754 else
755 puts("PASS");
756 }
757 else
758 {
759 puts("FAIL (returned 0)");
760 status ++;
761 }
762
763 ppdMarkOption(ppd, "Quality", "Photo");
764
765 fputs("ppdPageSizeLimits(Quality=Photo): ", stdout);
766 if (ppdPageSizeLimits(ppd, &minsize, &maxsize))
767 {
768 if (minsize.width != 200 || minsize.length != 200 ||
769 maxsize.width != 1000 || maxsize.length != 1000)
770 {
771 printf("FAIL (got min=%.0fx%.0f, max=%.0fx%.0f, "
772 "expected min=200x200, max=1000x1000)\n", minsize.width,
773 minsize.length, maxsize.width, maxsize.length);
774 status ++;
775 }
776 else
777 puts("PASS");
778 }
779 else
780 {
781 puts("FAIL (returned 0)");
782 status ++;
783 }
784
785 ppdMarkOption(ppd, "InputSlot", "Tray");
786
787 fputs("ppdPageSizeLimits(Quality=Photo): ", stdout);
788 if (ppdPageSizeLimits(ppd, &minsize, &maxsize))
789 {
790 if (minsize.width != 300 || minsize.length != 300 ||
791 maxsize.width != 1080 || maxsize.length != 86400)
792 {
793 printf("FAIL (got min=%.0fx%.0f, max=%.0fx%.0f, "
794 "expected min=300x300, max=1080x86400)\n", minsize.width,
795 minsize.length, maxsize.width, maxsize.length);
796 status ++;
797 }
798 else
799 puts("PASS");
800 }
801 else
802 {
803 puts("FAIL (returned 0)");
804 status ++;
805 }
806 }
807 else
808 {
809 const char *filename; /* PPD filename */
810 struct stat fileinfo; /* File information */
811
812
813 if (!strncmp(argv[1], "-d", 2))
814 {
815 filename = cupsGetPPD(argv[1] + 2);
816 if (!filename)
817 {
818 printf("%s: %s\n", argv[1], cupsLastErrorString());
819 return (1);
820 }
821 }
822 else
823 filename = argv[1];
824
825 if (lstat(filename, &fileinfo))
826 {
827 printf("%s: %s\n", filename, strerror(errno));
828 return (1);
829 }
830
831 if (S_ISLNK(fileinfo.st_mode))
832 {
833 char realfile[1024]; /* Real file path */
834 ssize_t realsize; /* Size of real file path */
835
836
837 if ((realsize = readlink(filename, realfile, sizeof(realfile) - 1)) < 0)
838 strcpy(realfile, "Unknown");
839 else
840 realfile[realsize] = '\0';
841
842 if (stat(realfile, &fileinfo))
843 printf("%s: symlink to \"%s\", %s\n", filename, realfile,
844 strerror(errno));
845 else
846 printf("%s: symlink to \"%s\", %ld bytes\n", filename, realfile,
847 (long)fileinfo.st_size);
848 }
849 else
850 printf("%s: regular file, %ld bytes\n", filename, (long)fileinfo.st_size);
851
852 if ((ppd = ppdOpenFile(filename)) == NULL)
853 {
854 ppd_status_t err; /* Last error in file */
855 int line; /* Line number in file */
856
857
858 status ++;
859 err = ppdLastError(&line);
860
861 printf("%s: %s on line %d\n", argv[1], ppdErrorString(err), line);
862 }
863 else
864 {
865 int j, k; /* Looping vars */
866 ppd_group_t *group; /* Option group */
867 ppd_option_t *option; /* Option */
868 ppd_coption_t *coption; /* Custom option */
869 ppd_cparam_t *cparam; /* Custom parameter */
870 ppd_const_t *c; /* UIConstraints */
871 char lang[255], /* LANG environment variable */
872 lc_all[255], /* LC_ALL environment variable */
873 lc_ctype[255], /* LC_CTYPE environment variable */
874 lc_messages[255];/* LC_MESSAGES environment variable */
875
876
877 if (argc > 2)
878 {
879 snprintf(lang, sizeof(lang), "LANG=%s", argv[2]);
880 putenv(lang);
881 snprintf(lc_all, sizeof(lc_all), "LC_ALL=%s", argv[2]);
882 putenv(lc_all);
883 snprintf(lc_ctype, sizeof(lc_ctype), "LC_CTYPE=%s", argv[2]);
884 putenv(lc_ctype);
885 snprintf(lc_messages, sizeof(lc_messages), "LC_MESSAGES=%s", argv[2]);
886 putenv(lc_messages);
887 }
888
889 ppdLocalize(ppd);
890 ppdMarkDefaults(ppd);
891
892 if (argc > 3)
893 {
894 text = ppdLocalizeIPPReason(ppd, argv[3], NULL, buffer, sizeof(buffer));
895 printf("ppdLocalizeIPPReason(%s)=%s\n", argv[3],
896 text ? text : "(null)");
897 return (text == NULL);
898 }
899
900 for (i = ppd->num_groups, group = ppd->groups;
901 i > 0;
902 i --, group ++)
903 {
904 printf("%s (%s):\n", group->name, group->text);
905
906 for (j = group->num_options, option = group->options;
907 j > 0;
908 j --, option ++)
909 {
910 printf(" %s (%s):\n", option->keyword, option->text);
911
912 for (k = 0; k < option->num_choices; k ++)
913 printf(" - %s%s (%s)\n",
914 option->choices[k].marked ? "*" : "",
915 option->choices[k].choice, option->choices[k].text);
916
917 if ((coption = ppdFindCustomOption(ppd, option->keyword)) != NULL)
918 {
919 for (cparam = (ppd_cparam_t *)cupsArrayFirst(coption->params);
920 cparam;
921 cparam = (ppd_cparam_t *)cupsArrayNext(coption->params))
922 {
923 switch (cparam->type)
924 {
925 case PPD_CUSTOM_CURVE :
926 printf(" %s(%s): PPD_CUSTOM_CURVE (%g to %g)\n",
927 cparam->name, cparam->text,
928 cparam->minimum.custom_curve,
929 cparam->maximum.custom_curve);
930 break;
931
932 case PPD_CUSTOM_INT :
933 printf(" %s(%s): PPD_CUSTOM_INT (%d to %d)\n",
934 cparam->name, cparam->text,
935 cparam->minimum.custom_int,
936 cparam->maximum.custom_int);
937 break;
938
939 case PPD_CUSTOM_INVCURVE :
940 printf(" %s(%s): PPD_CUSTOM_INVCURVE (%g to %g)\n",
941 cparam->name, cparam->text,
942 cparam->minimum.custom_invcurve,
943 cparam->maximum.custom_invcurve);
944 break;
945
946 case PPD_CUSTOM_PASSCODE :
947 printf(" %s(%s): PPD_CUSTOM_PASSCODE (%d to %d)\n",
948 cparam->name, cparam->text,
949 cparam->minimum.custom_passcode,
950 cparam->maximum.custom_passcode);
951 break;
952
953 case PPD_CUSTOM_PASSWORD :
954 printf(" %s(%s): PPD_CUSTOM_PASSWORD (%d to %d)\n",
955 cparam->name, cparam->text,
956 cparam->minimum.custom_password,
957 cparam->maximum.custom_password);
958 break;
959
960 case PPD_CUSTOM_POINTS :
961 printf(" %s(%s): PPD_CUSTOM_POINTS (%g to %g)\n",
962 cparam->name, cparam->text,
963 cparam->minimum.custom_points,
964 cparam->maximum.custom_points);
965 break;
966
967 case PPD_CUSTOM_REAL :
968 printf(" %s(%s): PPD_CUSTOM_REAL (%g to %g)\n",
969 cparam->name, cparam->text,
970 cparam->minimum.custom_real,
971 cparam->maximum.custom_real);
972 break;
973
974 case PPD_CUSTOM_STRING :
975 printf(" %s(%s): PPD_CUSTOM_STRING (%d to %d)\n",
976 cparam->name, cparam->text,
977 cparam->minimum.custom_string,
978 cparam->maximum.custom_string);
979 break;
980 }
981 }
982 }
983 }
984 }
985
986 puts("\nConstraints:");
987
988 for (i = ppd->num_consts, c = ppd->consts; i > 0; i --, c ++)
989 printf(" *UIConstraints: *%s %s *%s %s\n", c->option1, c->choice1,
990 c->option2, c->choice2);
991 if (ppd->num_consts == 0)
992 puts(" NO CONSTRAINTS");
993
994 puts("\nFilters:");
995
996 for (i = 0; i < ppd->num_filters; i ++)
997 printf(" %s\n", ppd->filters[i]);
998
999 if (ppd->num_filters == 0)
1000 puts(" NO FILTERS");
1001
1002 puts("\nAttributes:");
1003
1004 for (attr = (ppd_attr_t *)cupsArrayFirst(ppd->sorted_attrs);
1005 attr;
1006 attr = (ppd_attr_t *)cupsArrayNext(ppd->sorted_attrs))
1007 printf(" *%s %s/%s: \"%s\"\n", attr->name, attr->spec,
1008 attr->text, attr->value ? attr->value : "");
1009 }
1010
1011 if (!strncmp(argv[1], "-d", 2))
1012 unlink(filename);
1013 }
1014
1015 #ifdef __APPLE__
1016 if (getenv("MallocStackLogging") && getenv("MallocStackLoggingNoCompact"))
1017 {
1018 char command[1024]; /* malloc_history command */
1019
1020 snprintf(command, sizeof(command), "malloc_history %d -all_by_size",
1021 getpid());
1022 fflush(stdout);
1023 system(command);
1024 }
1025 #endif /* __APPLE__ */
1026
1027 ppdClose(ppd);
1028
1029 return (status);
1030 }
1031
1032
1033 /*
1034 * End of "$Id: testppd.c 7897 2008-09-02 19:33:19Z mike $".
1035 */