1 /*
2 * Destination option/media support for CUPS.
3 *
4 * Copyright 2012-2016 by Apple Inc.
5 *
6 * These coded instructions, statements, and computer programs are the
7 * property of Apple Inc. and are protected by Federal copyright
8 * law. Distribution and use rights are outlined in the file "LICENSE.txt"
9 * which should have been included with this file. If this file is
10 * missing or damaged, see the license at "http://www.cups.org/".
11 *
12 * This file is subject to the Apple OS-Developed Software exception.
13 */
14
15 /*
16 * Include necessary headers...
17 */
18
19 #include "cups-private.h"
20
21
22 /*
23 * Local constants...
24 */
25
26 #define _CUPS_MEDIA_READY_TTL 30 /* Life of xxx-ready values */
27
28
29 /*
30 * Local functions...
31 */
32
33 static void cups_add_dconstres(cups_array_t *a, ipp_t *collection);
34 static int cups_compare_dconstres(_cups_dconstres_t *a,
35 _cups_dconstres_t *b);
36 static int cups_compare_media_db(_cups_media_db_t *a,
37 _cups_media_db_t *b);
38 static _cups_media_db_t *cups_copy_media_db(_cups_media_db_t *mdb);
39 static void cups_create_cached(http_t *http, cups_dinfo_t *dinfo,
40 unsigned flags);
41 static void cups_create_constraints(cups_dinfo_t *dinfo);
42 static void cups_create_defaults(cups_dinfo_t *dinfo);
43 static void cups_create_media_db(cups_dinfo_t *dinfo,
44 unsigned flags);
45 static void cups_free_media_db(_cups_media_db_t *mdb);
46 static int cups_get_media_db(http_t *http, cups_dinfo_t *dinfo,
47 pwg_media_t *pwg, unsigned flags,
48 cups_size_t *size);
49 static int cups_is_close_media_db(_cups_media_db_t *a,
50 _cups_media_db_t *b);
51 static cups_array_t *cups_test_constraints(cups_dinfo_t *dinfo,
52 const char *new_option,
53 const char *new_value,
54 int num_options,
55 cups_option_t *options,
56 int *num_conflicts,
57 cups_option_t **conflicts);
58 static void cups_update_ready(http_t *http, cups_dinfo_t *dinfo);
59
60
61 /*
62 * 'cupsCheckDestSupported()' - Check that the option and value are supported
63 * by the destination.
64 *
65 * Returns 1 if supported, 0 otherwise.
66 *
67 * @since CUPS 1.6/macOS 10.8@
68 */
69
70 int /* O - 1 if supported, 0 otherwise */
cupsCheckDestSupported(http_t * http,cups_dest_t * dest,cups_dinfo_t * dinfo,const char * option,const char * value)71 cupsCheckDestSupported(
72 http_t *http, /* I - Connection to destination */
73 cups_dest_t *dest, /* I - Destination */
74 cups_dinfo_t *dinfo, /* I - Destination information */
75 const char *option, /* I - Option */
76 const char *value) /* I - Value */
77 {
78 int i; /* Looping var */
79 char temp[1024]; /* Temporary string */
80 int int_value; /* Integer value */
81 int xres_value, /* Horizontal resolution */
82 yres_value; /* Vertical resolution */
83 ipp_res_t units_value; /* Resolution units */
84 ipp_attribute_t *attr; /* Attribute */
85 _ipp_value_t *attrval; /* Current attribute value */
86
87
88 /*
89 * Range check input...
90 */
91
92 if (!http || !dest || !dinfo || !option || !value)
93 return (0);
94
95 /*
96 * Lookup the attribute...
97 */
98
99 if (strstr(option, "-supported"))
100 attr = ippFindAttribute(dinfo->attrs, option, IPP_TAG_ZERO);
101 else
102 {
103 snprintf(temp, sizeof(temp), "%s-supported", option);
104 attr = ippFindAttribute(dinfo->attrs, temp, IPP_TAG_ZERO);
105 }
106
107 if (!attr)
108 return (0);
109
110 /*
111 * Compare values...
112 */
113
114 if (!strcmp(option, "media") && !strncmp(value, "custom_", 7))
115 {
116 /*
117 * Check range of custom media sizes...
118 */
119
120 pwg_media_t *pwg; /* Current PWG media size info */
121 int min_width, /* Minimum width */
122 min_length, /* Minimum length */
123 max_width, /* Maximum width */
124 max_length; /* Maximum length */
125
126 /*
127 * Get the minimum and maximum size...
128 */
129
130 min_width = min_length = INT_MAX;
131 max_width = max_length = 0;
132
133 for (i = attr->num_values, attrval = attr->values;
134 i > 0;
135 i --, attrval ++)
136 {
137 if (!strncmp(attrval->string.text, "custom_min_", 11) &&
138 (pwg = pwgMediaForPWG(attrval->string.text)) != NULL)
139 {
140 min_width = pwg->width;
141 min_length = pwg->length;
142 }
143 else if (!strncmp(attrval->string.text, "custom_max_", 11) &&
144 (pwg = pwgMediaForPWG(attrval->string.text)) != NULL)
145 {
146 max_width = pwg->width;
147 max_length = pwg->length;
148 }
149 }
150
151 /*
152 * Check the range...
153 */
154
155 if (min_width < INT_MAX && max_width > 0 &&
156 (pwg = pwgMediaForPWG(value)) != NULL &&
157 pwg->width >= min_width && pwg->width <= max_width &&
158 pwg->length >= min_length && pwg->length <= max_length)
159 return (1);
160 }
161 else
162 {
163 /*
164 * Check literal values...
165 */
166
167 switch (attr->value_tag)
168 {
169 case IPP_TAG_INTEGER :
170 case IPP_TAG_ENUM :
171 int_value = atoi(value);
172
173 for (i = 0; i < attr->num_values; i ++)
174 if (attr->values[i].integer == int_value)
175 return (1);
176 break;
177
178 case IPP_TAG_BOOLEAN :
179 return (attr->values[0].boolean);
180
181 case IPP_TAG_RANGE :
182 int_value = atoi(value);
183
184 for (i = 0; i < attr->num_values; i ++)
185 if (int_value >= attr->values[i].range.lower &&
186 int_value <= attr->values[i].range.upper)
187 return (1);
188 break;
189
190 case IPP_TAG_RESOLUTION :
191 if (sscanf(value, "%dx%d%15s", &xres_value, &yres_value, temp) != 3)
192 {
193 if (sscanf(value, "%d%15s", &xres_value, temp) != 2)
194 return (0);
195
196 yres_value = xres_value;
197 }
198
199 if (!strcmp(temp, "dpi"))
200 units_value = IPP_RES_PER_INCH;
201 else if (!strcmp(temp, "dpc") || !strcmp(temp, "dpcm"))
202 units_value = IPP_RES_PER_CM;
203 else
204 return (0);
205
206 for (i = attr->num_values, attrval = attr->values;
207 i > 0;
208 i --, attrval ++)
209 {
210 if (attrval->resolution.xres == xres_value &&
211 attrval->resolution.yres == yres_value &&
212 attrval->resolution.units == units_value)
213 return (1);
214 }
215 break;
216
217 case IPP_TAG_TEXT :
218 case IPP_TAG_NAME :
219 case IPP_TAG_KEYWORD :
220 case IPP_TAG_CHARSET :
221 case IPP_TAG_URI :
222 case IPP_TAG_URISCHEME :
223 case IPP_TAG_MIMETYPE :
224 case IPP_TAG_LANGUAGE :
225 case IPP_TAG_TEXTLANG :
226 case IPP_TAG_NAMELANG :
227 for (i = 0; i < attr->num_values; i ++)
228 if (!strcmp(attr->values[i].string.text, value))
229 return (1);
230 break;
231
232 default :
233 break;
234 }
235 }
236
237 /*
238 * If we get there the option+value is not supported...
239 */
240
241 return (0);
242 }
243
244
245 /*
246 * 'cupsCopyDestConflicts()' - Get conflicts and resolutions for a new
247 * option/value pair.
248 *
249 * "num_options" and "options" represent the currently selected options by the
250 * user. "new_option" and "new_value" are the setting the user has just
251 * changed.
252 *
253 * Returns 1 if there is a conflict, 0 if there are no conflicts, and -1 if
254 * there was an unrecoverable error such as a resolver loop.
255 *
256 * If "num_conflicts" and "conflicts" are not @code NULL@, they are set to
257 * contain the list of conflicting option/value pairs. Similarly, if
258 * "num_resolved" and "resolved" are not @code NULL@ they will be set to the
259 * list of changes needed to resolve the conflict.
260 *
261 * If cupsCopyDestConflicts returns 1 but "num_resolved" and "resolved" are set
262 * to 0 and @code NULL@, respectively, then the conflict cannot be resolved.
263 *
264 * @since CUPS 1.6/macOS 10.8@
265 */
266
267 int /* O - 1 if there is a conflict, 0 if none, -1 on error */
cupsCopyDestConflicts(http_t * http,cups_dest_t * dest,cups_dinfo_t * dinfo,int num_options,cups_option_t * options,const char * new_option,const char * new_value,int * num_conflicts,cups_option_t ** conflicts,int * num_resolved,cups_option_t ** resolved)268 cupsCopyDestConflicts(
269 http_t *http, /* I - Connection to destination */
270 cups_dest_t *dest, /* I - Destination */
271 cups_dinfo_t *dinfo, /* I - Destination information */
272 int num_options, /* I - Number of current options */
273 cups_option_t *options, /* I - Current options */
274 const char *new_option, /* I - New option */
275 const char *new_value, /* I - New value */
276 int *num_conflicts, /* O - Number of conflicting options */
277 cups_option_t **conflicts, /* O - Conflicting options */
278 int *num_resolved, /* O - Number of options to resolve */
279 cups_option_t **resolved) /* O - Resolved options */
280 {
281 int i, /* Looping var */
282 have_conflicts = 0, /* Do we have conflicts? */
283 changed, /* Did we change something? */
284 tries, /* Number of tries for resolution */
285 num_myconf = 0, /* My number of conflicting options */
286 num_myres = 0; /* My number of resolved options */
287 cups_option_t *myconf = NULL, /* My conflicting options */
288 *myres = NULL, /* My resolved options */
289 *myoption, /* My current option */
290 *option; /* Current option */
291 cups_array_t *active = NULL, /* Active conflicts */
292 *pass = NULL, /* Resolvers for this pass */
293 *resolvers = NULL, /* Resolvers we have used */
294 *test; /* Test array for conflicts */
295 _cups_dconstres_t *c, /* Current constraint */
296 *r; /* Current resolver */
297 ipp_attribute_t *attr; /* Current attribute */
298 char value[2048]; /* Current attribute value as string */
299 const char *myvalue; /* Current value of an option */
300
301
302 /*
303 * Clear returned values...
304 */
305
306 if (num_conflicts)
307 *num_conflicts = 0;
308
309 if (conflicts)
310 *conflicts = NULL;
311
312 if (num_resolved)
313 *num_resolved = 0;
314
315 if (resolved)
316 *resolved = NULL;
317
318 /*
319 * Range check input...
320 */
321
322 if (!http || !dest || !dinfo ||
323 (num_conflicts != NULL) != (conflicts != NULL) ||
324 (num_resolved != NULL) != (resolved != NULL))
325 return (0);
326
327 /*
328 * Load constraints as needed...
329 */
330
331 if (!dinfo->constraints)
332 cups_create_constraints(dinfo);
333
334 if (cupsArrayCount(dinfo->constraints) == 0)
335 return (0);
336
337 if (!dinfo->num_defaults)
338 cups_create_defaults(dinfo);
339
340 /*
341 * If we are resolving, create a shadow array...
342 */
343
344 if (num_resolved)
345 {
346 for (i = num_options, option = options; i > 0; i --, option ++)
347 num_myres = cupsAddOption(option->name, option->value, num_myres, &myres);
348
349 if (new_option && new_value)
350 num_myres = cupsAddOption(new_option, new_value, num_myres, &myres);
351 }
352 else
353 {
354 num_myres = num_options;
355 myres = options;
356 }
357
358 /*
359 * Check for any conflicts...
360 */
361
362 if (num_resolved)
363 pass = cupsArrayNew((cups_array_func_t)cups_compare_dconstres, NULL);
364
365 for (tries = 0; tries < 100; tries ++)
366 {
367 /*
368 * Check for any conflicts...
369 */
370
371 if (num_conflicts || num_resolved)
372 {
373 cupsFreeOptions(num_myconf, myconf);
374
375 num_myconf = 0;
376 myconf = NULL;
377 active = cups_test_constraints(dinfo, new_option, new_value,
378 num_myres, myres, &num_myconf,
379 &myconf);
380 }
381 else
382 active = cups_test_constraints(dinfo, new_option, new_value, num_myres,
383 myres, NULL, NULL);
384
385 have_conflicts = (active != NULL);
386
387 if (!active || !num_resolved)
388 break; /* All done */
389
390 /*
391 * Scan the constraints that were triggered to apply resolvers...
392 */
393
394 if (!resolvers)
395 resolvers = cupsArrayNew((cups_array_func_t)cups_compare_dconstres, NULL);
396
397 for (c = (_cups_dconstres_t *)cupsArrayFirst(active), changed = 0;
398 c;
399 c = (_cups_dconstres_t *)cupsArrayNext(active))
400 {
401 if (cupsArrayFind(pass, c))
402 continue; /* Already applied this resolver... */
403
404 if (cupsArrayFind(resolvers, c))
405 {
406 DEBUG_printf(("1cupsCopyDestConflicts: Resolver loop with %s.",
407 c->name));
408 have_conflicts = -1;
409 goto cleanup;
410 }
411
412 if ((r = cupsArrayFind(dinfo->resolvers, c)) == NULL)
413 {
414 DEBUG_printf(("1cupsCopyDestConflicts: Resolver %s not found.",
415 c->name));
416 have_conflicts = -1;
417 goto cleanup;
418 }
419
420 /*
421 * Add the options from the resolver...
422 */
423
424 cupsArrayAdd(pass, r);
425 cupsArrayAdd(resolvers, r);
426
427 for (attr = ippFirstAttribute(r->collection);
428 attr;
429 attr = ippNextAttribute(r->collection))
430 {
431 if (new_option && !strcmp(attr->name, new_option))
432 continue; /* Ignore this if we just changed it */
433
434 if (ippAttributeString(attr, value, sizeof(value)) >= sizeof(value))
435 continue; /* Ignore if the value is too long */
436
437 if ((test = cups_test_constraints(dinfo, attr->name, value, num_myres,
438 myres, NULL, NULL)) == NULL)
439 {
440 /*
441 * That worked, flag it...
442 */
443
444 changed = 1;
445 }
446 else
447 cupsArrayDelete(test);
448
449 /*
450 * Add the option/value from the resolver regardless of whether it
451 * worked; this makes sure that we can cascade several changes to
452 * make things resolve...
453 */
454
455 num_myres = cupsAddOption(attr->name, value, num_myres, &myres);
456 }
457 }
458
459 if (!changed)
460 {
461 DEBUG_puts("1cupsCopyDestConflicts: Unable to resolve constraints.");
462 have_conflicts = -1;
463 goto cleanup;
464 }
465
466 cupsArrayClear(pass);
467
468 cupsArrayDelete(active);
469 active = NULL;
470 }
471
472 if (tries >= 100)
473 {
474 DEBUG_puts("1cupsCopyDestConflicts: Unable to resolve after 100 tries.");
475 have_conflicts = -1;
476 goto cleanup;
477 }
478
479 /*
480 * Copy resolved options as needed...
481 */
482
483 if (num_resolved)
484 {
485 for (i = num_myres, myoption = myres; i > 0; i --, myoption ++)
486 {
487 if ((myvalue = cupsGetOption(myoption->name, num_options,
488 options)) == NULL ||
489 strcmp(myvalue, myoption->value))
490 {
491 if (new_option && !strcmp(new_option, myoption->name) &&
492 new_value && !strcmp(new_value, myoption->value))
493 continue;
494
495 *num_resolved = cupsAddOption(myoption->name, myoption->value,
496 *num_resolved, resolved);
497 }
498 }
499 }
500
501 /*
502 * Clean up...
503 */
504
505 cleanup:
506
507 cupsArrayDelete(active);
508 cupsArrayDelete(pass);
509 cupsArrayDelete(resolvers);
510
511 if (num_resolved)
512 {
513 /*
514 * Free shadow copy of options...
515 */
516
517 cupsFreeOptions(num_myres, myres);
518 }
519
520 if (num_conflicts)
521 {
522 /*
523 * Return conflicting options to caller...
524 */
525
526 *num_conflicts = num_myconf;
527 *conflicts = myconf;
528 }
529 else
530 {
531 /*
532 * Free conflicting options...
533 */
534
535 cupsFreeOptions(num_myconf, myconf);
536 }
537
538 return (have_conflicts);
539 }
540
541
542 /*
543 * 'cupsCopyDestInfo()' - Get the supported values/capabilities for the
544 * destination.
545 *
546 * The caller is responsible for calling @link cupsFreeDestInfo@ on the return
547 * value. @code NULL@ is returned on error.
548 *
549 * @since CUPS 1.6/macOS 10.8@
550 */
551
552 cups_dinfo_t * /* O - Destination information */
cupsCopyDestInfo(http_t * http,cups_dest_t * dest)553 cupsCopyDestInfo(
554 http_t *http, /* I - Connection to destination */
555 cups_dest_t *dest) /* I - Destination */
556 {
557 cups_dinfo_t *dinfo; /* Destination information */
558 ipp_t *request, /* Get-Printer-Attributes request */
559 *response; /* Supported attributes */
560 int tries, /* Number of tries so far */
561 delay, /* Current retry delay */
562 prev_delay; /* Next retry delay */
563 const char *uri; /* Printer URI */
564 char resource[1024]; /* Resource path */
565 int version; /* IPP version */
566 ipp_status_t status; /* Status of request */
567 static const char * const requested_attrs[] =
568 { /* Requested attributes */
569 "job-template",
570 "media-col-database",
571 "printer-description"
572 };
573
574
575 DEBUG_printf(("cupsCopyDestSupported(http=%p, dest=%p(%s))", (void *)http, (void *)dest, dest ? dest->name : ""));
576
577 /*
578 * Range check input...
579 */
580
581 if (!http || !dest)
582 return (NULL);
583
584 /*
585 * Get the printer URI and resource path...
586 */
587
588 if ((uri = _cupsGetDestResource(dest, resource, sizeof(resource))) == NULL)
589 return (NULL);
590
591 /*
592 * Get the supported attributes...
593 */
594
595 delay = 1;
596 prev_delay = 1;
597 tries = 0;
598 version = 20;
599
600 do
601 {
602 /*
603 * Send a Get-Printer-Attributes request...
604 */
605
606 request = ippNewRequest(IPP_OP_GET_PRINTER_ATTRIBUTES);
607 ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", NULL,
608 uri);
609 ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME,
610 "requesting-user-name", NULL, cupsUser());
611 ippAddStrings(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD,
612 "requested-attributes",
613 (int)(sizeof(requested_attrs) / sizeof(requested_attrs[0])),
614 NULL, requested_attrs);
615 response = cupsDoRequest(http, request, resource);
616 status = cupsLastError();
617
618 if (status > IPP_STATUS_OK_IGNORED_OR_SUBSTITUTED)
619 {
620 DEBUG_printf(("cupsCopyDestSupported: Get-Printer-Attributes for '%s' "
621 "returned %s (%s)", dest->name, ippErrorString(status),
622 cupsLastErrorString()));
623
624 ippDelete(response);
625 response = NULL;
626
627 if (status == IPP_STATUS_ERROR_VERSION_NOT_SUPPORTED && version > 11)
628 version = 11;
629 else if (status == IPP_STATUS_ERROR_BUSY)
630 {
631 sleep((unsigned)delay);
632
633 delay = _cupsNextDelay(delay, &prev_delay);
634 }
635 else
636 return (NULL);
637 }
638
639 tries ++;
640 }
641 while (!response && tries < 10);
642
643 if (!response)
644 return (NULL);
645
646 /*
647 * Allocate a cups_dinfo_t structure and return it...
648 */
649
650 if ((dinfo = calloc(1, sizeof(cups_dinfo_t))) == NULL)
651 {
652 _cupsSetError(IPP_STATUS_ERROR_INTERNAL, strerror(errno), 0);
653 ippDelete(response);
654 return (NULL);
655 }
656
657 dinfo->version = version;
658 dinfo->uri = uri;
659 dinfo->resource = _cupsStrAlloc(resource);
660 dinfo->attrs = response;
661
662 return (dinfo);
663 }
664
665
666 /*
667 * 'cupsFindDestDefault()' - Find the default value(s) for the given option.
668 *
669 * The returned value is an IPP attribute. Use the @code ippGetBoolean@,
670 * @code ippGetCollection@, @code ippGetCount@, @code ippGetDate@,
671 * @code ippGetInteger@, @code ippGetOctetString@, @code ippGetRange@,
672 * @code ippGetResolution@, @code ippGetString@, and @code ippGetValueTag@
673 * functions to inspect the default value(s) as needed.
674 *
675 * @since CUPS 1.7/macOS 10.9@
676 */
677
678 ipp_attribute_t * /* O - Default attribute or @code NULL@ for none */
cupsFindDestDefault(http_t * http,cups_dest_t * dest,cups_dinfo_t * dinfo,const char * option)679 cupsFindDestDefault(
680 http_t *http, /* I - Connection to destination */
681 cups_dest_t *dest, /* I - Destination */
682 cups_dinfo_t *dinfo, /* I - Destination information */
683 const char *option) /* I - Option/attribute name */
684 {
685 char name[IPP_MAX_NAME]; /* Attribute name */
686
687
688 /*
689 * Range check input...
690 */
691
692 if (!http || !dest || !dinfo || !option)
693 {
694 _cupsSetError(IPP_STATUS_ERROR_INTERNAL, strerror(EINVAL), 0);
695 return (NULL);
696 }
697
698 /*
699 * Find and return the attribute...
700 */
701
702 snprintf(name, sizeof(name), "%s-default", option);
703 return (ippFindAttribute(dinfo->attrs, name, IPP_TAG_ZERO));
704 }
705
706
707 /*
708 * 'cupsFindDestReady()' - Find the default value(s) for the given option.
709 *
710 * The returned value is an IPP attribute. Use the @code ippGetBoolean@,
711 * @code ippGetCollection@, @code ippGetCount@, @code ippGetDate@,
712 * @code ippGetInteger@, @code ippGetOctetString@, @code ippGetRange@,
713 * @code ippGetResolution@, @code ippGetString@, and @code ippGetValueTag@
714 * functions to inspect the default value(s) as needed.
715 *
716 * @since CUPS 1.7/macOS 10.9@
717 */
718
719 ipp_attribute_t * /* O - Default attribute or @code NULL@ for none */
cupsFindDestReady(http_t * http,cups_dest_t * dest,cups_dinfo_t * dinfo,const char * option)720 cupsFindDestReady(
721 http_t *http, /* I - Connection to destination */
722 cups_dest_t *dest, /* I - Destination */
723 cups_dinfo_t *dinfo, /* I - Destination information */
724 const char *option) /* I - Option/attribute name */
725 {
726 char name[IPP_MAX_NAME]; /* Attribute name */
727
728
729 /*
730 * Range check input...
731 */
732
733 if (!http || !dest || !dinfo || !option)
734 {
735 _cupsSetError(IPP_STATUS_ERROR_INTERNAL, strerror(EINVAL), 0);
736 return (NULL);
737 }
738
739 /*
740 * Find and return the attribute...
741 */
742
743 cups_update_ready(http, dinfo);
744
745 snprintf(name, sizeof(name), "%s-ready", option);
746 return (ippFindAttribute(dinfo->ready_attrs, name, IPP_TAG_ZERO));
747 }
748
749
750 /*
751 * 'cupsFindDestSupported()' - Find the default value(s) for the given option.
752 *
753 * The returned value is an IPP attribute. Use the @code ippGetBoolean@,
754 * @code ippGetCollection@, @code ippGetCount@, @code ippGetDate@,
755 * @code ippGetInteger@, @code ippGetOctetString@, @code ippGetRange@,
756 * @code ippGetResolution@, @code ippGetString@, and @code ippGetValueTag@
757 * functions to inspect the default value(s) as needed.
758 *
759 * @since CUPS 1.7/macOS 10.9@
760 */
761
762 ipp_attribute_t * /* O - Default attribute or @code NULL@ for none */
cupsFindDestSupported(http_t * http,cups_dest_t * dest,cups_dinfo_t * dinfo,const char * option)763 cupsFindDestSupported(
764 http_t *http, /* I - Connection to destination */
765 cups_dest_t *dest, /* I - Destination */
766 cups_dinfo_t *dinfo, /* I - Destination information */
767 const char *option) /* I - Option/attribute name */
768 {
769 char name[IPP_MAX_NAME]; /* Attribute name */
770
771
772 /*
773 * Range check input...
774 */
775
776 if (!http || !dest || !dinfo || !option)
777 {
778 _cupsSetError(IPP_STATUS_ERROR_INTERNAL, strerror(EINVAL), 0);
779 return (NULL);
780 }
781
782 /*
783 * Find and return the attribute...
784 */
785
786 snprintf(name, sizeof(name), "%s-supported", option);
787 return (ippFindAttribute(dinfo->attrs, name, IPP_TAG_ZERO));
788 }
789
790
791 /*
792 * 'cupsFreeDestInfo()' - Free destination information obtained using
793 * @link cupsCopyDestInfo@.
794 */
795
796 void
cupsFreeDestInfo(cups_dinfo_t * dinfo)797 cupsFreeDestInfo(cups_dinfo_t *dinfo) /* I - Destination information */
798 {
799 /*
800 * Range check input...
801 */
802
803 if (!dinfo)
804 return;
805
806 /*
807 * Free memory and return...
808 */
809
810 _cupsStrFree(dinfo->resource);
811
812 cupsArrayDelete(dinfo->constraints);
813 cupsArrayDelete(dinfo->resolvers);
814
815 cupsArrayDelete(dinfo->localizations);
816
817 cupsArrayDelete(dinfo->media_db);
818
819 cupsArrayDelete(dinfo->cached_db);
820
821 ippDelete(dinfo->ready_attrs);
822 cupsArrayDelete(dinfo->ready_db);
823
824 ippDelete(dinfo->attrs);
825
826 free(dinfo);
827 }
828
829
830 /*
831 * 'cupsGetDestMediaByIndex()' - Get a media name, dimension, and margins for a
832 * specific size.
833 *
834 * The @code flags@ parameter determines which set of media are indexed. For
835 * example, passing @code CUPS_MEDIA_FLAGS_BORDERLESS@ will get the Nth
836 * borderless size supported by the printer.
837 *
838 * @since CUPS 1.7/macOS 10.9@
839 */
840
841 int /* O - 1 on success, 0 on failure */
cupsGetDestMediaByIndex(http_t * http,cups_dest_t * dest,cups_dinfo_t * dinfo,int n,unsigned flags,cups_size_t * size)842 cupsGetDestMediaByIndex(
843 http_t *http, /* I - Connection to destination */
844 cups_dest_t *dest, /* I - Destination */
845 cups_dinfo_t *dinfo, /* I - Destination information */
846 int n, /* I - Media size number (0-based) */
847 unsigned flags, /* I - Media flags */
848 cups_size_t *size) /* O - Media size information */
849 {
850 _cups_media_db_t *nsize; /* Size for N */
851 pwg_media_t *pwg; /* PWG media name for size */
852
853
854 /*
855 * Range check input...
856 */
857
858 if (size)
859 memset(size, 0, sizeof(cups_size_t));
860
861 if (!http || !dest || !dinfo || n < 0 || !size)
862 {
863 _cupsSetError(IPP_STATUS_ERROR_INTERNAL, strerror(EINVAL), 0);
864 return (0);
865 }
866
867 /*
868 * Load media list as needed...
869 */
870
871 if (flags & CUPS_MEDIA_FLAGS_READY)
872 cups_update_ready(http, dinfo);
873
874 if (!dinfo->cached_db || dinfo->cached_flags != flags)
875 cups_create_cached(http, dinfo, flags);
876
877 /*
878 * Copy the size over and return...
879 */
880
881 if ((nsize = (_cups_media_db_t *)cupsArrayIndex(dinfo->cached_db, n)) == NULL)
882 {
883 _cupsSetError(IPP_STATUS_ERROR_INTERNAL, strerror(EINVAL), 0);
884 return (0);
885 }
886
887 if (nsize->size_name)
888 strlcpy(size->media, nsize->size_name, sizeof(size->media));
889 else if (nsize->key)
890 strlcpy(size->media, nsize->key, sizeof(size->media));
891 else if ((pwg = pwgMediaForSize(nsize->width, nsize->length)) != NULL)
892 strlcpy(size->media, pwg->pwg, sizeof(size->media));
893 else
894 {
895 _cupsSetError(IPP_STATUS_ERROR_INTERNAL, strerror(EINVAL), 0);
896 return (0);
897 }
898
899 size->width = nsize->width;
900 size->length = nsize->length;
901 size->bottom = nsize->bottom;
902 size->left = nsize->left;
903 size->right = nsize->right;
904 size->top = nsize->top;
905
906 return (1);
907 }
908
909
910 /*
911 * 'cupsGetDestMediaByName()' - Get media names, dimensions, and margins.
912 *
913 * The "media" string is a PWG media name. "Flags" provides some matching
914 * guidance (multiple flags can be combined):
915 *
916 * CUPS_MEDIA_FLAGS_DEFAULT = find the closest size supported by the printer,
917 * CUPS_MEDIA_FLAGS_BORDERLESS = find a borderless size,
918 * CUPS_MEDIA_FLAGS_DUPLEX = find a size compatible with 2-sided printing,
919 * CUPS_MEDIA_FLAGS_EXACT = find an exact match for the size, and
920 * CUPS_MEDIA_FLAGS_READY = if the printer supports media sensing, find the
921 * size amongst the "ready" media.
922 *
923 * The matching result (if any) is returned in the "cups_size_t" structure.
924 *
925 * Returns 1 when there is a match and 0 if there is not a match.
926 *
927 * @since CUPS 1.6/macOS 10.8@
928 */
929
930 int /* O - 1 on match, 0 on failure */
cupsGetDestMediaByName(http_t * http,cups_dest_t * dest,cups_dinfo_t * dinfo,const char * media,unsigned flags,cups_size_t * size)931 cupsGetDestMediaByName(
932 http_t *http, /* I - Connection to destination */
933 cups_dest_t *dest, /* I - Destination */
934 cups_dinfo_t *dinfo, /* I - Destination information */
935 const char *media, /* I - Media name */
936 unsigned flags, /* I - Media matching flags */
937 cups_size_t *size) /* O - Media size information */
938 {
939 pwg_media_t *pwg; /* PWG media info */
940
941
942 /*
943 * Range check input...
944 */
945
946 if (size)
947 memset(size, 0, sizeof(cups_size_t));
948
949 if (!http || !dest || !dinfo || !media || !size)
950 {
951 _cupsSetError(IPP_STATUS_ERROR_INTERNAL, strerror(EINVAL), 0);
952 return (0);
953 }
954
955 /*
956 * Lookup the media size name...
957 */
958
959 if ((pwg = pwgMediaForPWG(media)) == NULL)
960 if ((pwg = pwgMediaForLegacy(media)) == NULL)
961 {
962 DEBUG_printf(("1cupsGetDestMediaByName: Unknown size '%s'.", media));
963 _cupsSetError(IPP_STATUS_ERROR_INTERNAL, _("Unknown media size name."), 1);
964 return (0);
965 }
966
967 /*
968 * Lookup the size...
969 */
970
971 return (cups_get_media_db(http, dinfo, pwg, flags, size));
972 }
973
974
975 /*
976 * 'cupsGetDestMediaBySize()' - Get media names, dimensions, and margins.
977 *
978 * "Width" and "length" are the dimensions in hundredths of millimeters.
979 * "Flags" provides some matching guidance (multiple flags can be combined):
980 *
981 * CUPS_MEDIA_FLAGS_DEFAULT = find the closest size supported by the printer,
982 * CUPS_MEDIA_FLAGS_BORDERLESS = find a borderless size,
983 * CUPS_MEDIA_FLAGS_DUPLEX = find a size compatible with 2-sided printing,
984 * CUPS_MEDIA_FLAGS_EXACT = find an exact match for the size, and
985 * CUPS_MEDIA_FLAGS_READY = if the printer supports media sensing, find the
986 * size amongst the "ready" media.
987 *
988 * The matching result (if any) is returned in the "cups_size_t" structure.
989 *
990 * Returns 1 when there is a match and 0 if there is not a match.
991 *
992 * @since CUPS 1.6/macOS 10.8@
993 */
994
995 int /* O - 1 on match, 0 on failure */
cupsGetDestMediaBySize(http_t * http,cups_dest_t * dest,cups_dinfo_t * dinfo,int width,int length,unsigned flags,cups_size_t * size)996 cupsGetDestMediaBySize(
997 http_t *http, /* I - Connection to destination */
998 cups_dest_t *dest, /* I - Destination */
999 cups_dinfo_t *dinfo, /* I - Destination information */
1000 int width, /* I - Media width in hundredths of
1001 * of millimeters */
1002 int length, /* I - Media length in hundredths of
1003 * of millimeters */
1004 unsigned flags, /* I - Media matching flags */
1005 cups_size_t *size) /* O - Media size information */
1006 {
1007 pwg_media_t *pwg; /* PWG media info */
1008
1009
1010 /*
1011 * Range check input...
1012 */
1013
1014 if (size)
1015 memset(size, 0, sizeof(cups_size_t));
1016
1017 if (!http || !dest || !dinfo || width <= 0 || length <= 0 || !size)
1018 {
1019 _cupsSetError(IPP_STATUS_ERROR_INTERNAL, strerror(EINVAL), 0);
1020 return (0);
1021 }
1022
1023 /*
1024 * Lookup the media size name...
1025 */
1026
1027 if ((pwg = pwgMediaForSize(width, length)) == NULL)
1028 {
1029 DEBUG_printf(("1cupsGetDestMediaBySize: Invalid size %dx%d.", width,
1030 length));
1031 _cupsSetError(IPP_STATUS_ERROR_INTERNAL, _("Invalid media size."), 1);
1032 return (0);
1033 }
1034
1035 /*
1036 * Lookup the size...
1037 */
1038
1039 return (cups_get_media_db(http, dinfo, pwg, flags, size));
1040 }
1041
1042
1043 /*
1044 * 'cupsGetDestMediaCount()' - Get the number of sizes supported by a
1045 * destination.
1046 *
1047 * The @code flags@ parameter determines the set of media sizes that are
1048 * counted. For example, passing @code CUPS_MEDIA_FLAGS_BORDERLESS@ will return
1049 * the number of borderless sizes.
1050 *
1051 * @since CUPS 1.7/macOS 10.9@
1052 */
1053
1054 int /* O - Number of sizes */
cupsGetDestMediaCount(http_t * http,cups_dest_t * dest,cups_dinfo_t * dinfo,unsigned flags)1055 cupsGetDestMediaCount(
1056 http_t *http, /* I - Connection to destination */
1057 cups_dest_t *dest, /* I - Destination */
1058 cups_dinfo_t *dinfo, /* I - Destination information */
1059 unsigned flags) /* I - Media flags */
1060 {
1061 /*
1062 * Range check input...
1063 */
1064
1065 if (!http || !dest || !dinfo)
1066 {
1067 _cupsSetError(IPP_STATUS_ERROR_INTERNAL, strerror(EINVAL), 0);
1068 return (0);
1069 }
1070
1071 /*
1072 * Load media list as needed...
1073 */
1074
1075 if (flags & CUPS_MEDIA_FLAGS_READY)
1076 cups_update_ready(http, dinfo);
1077
1078 if (!dinfo->cached_db || dinfo->cached_flags != flags)
1079 cups_create_cached(http, dinfo, flags);
1080
1081 return (cupsArrayCount(dinfo->cached_db));
1082 }
1083
1084
1085 /*
1086 * 'cupsGetDestMediaDefault()' - Get the default size for a destination.
1087 *
1088 * The @code flags@ parameter determines which default size is returned. For
1089 * example, passing @code CUPS_MEDIA_FLAGS_BORDERLESS@ will return the default
1090 * borderless size, typically US Letter or A4, but sometimes 4x6 photo media.
1091 *
1092 * @since CUPS 1.7/macOS 10.9@
1093 */
1094
1095 int /* O - 1 on success, 0 on failure */
cupsGetDestMediaDefault(http_t * http,cups_dest_t * dest,cups_dinfo_t * dinfo,unsigned flags,cups_size_t * size)1096 cupsGetDestMediaDefault(
1097 http_t *http, /* I - Connection to destination */
1098 cups_dest_t *dest, /* I - Destination */
1099 cups_dinfo_t *dinfo, /* I - Destination information */
1100 unsigned flags, /* I - Media flags */
1101 cups_size_t *size) /* O - Media size information */
1102 {
1103 const char *media; /* Default media size */
1104
1105
1106 /*
1107 * Range check input...
1108 */
1109
1110 if (size)
1111 memset(size, 0, sizeof(cups_size_t));
1112
1113 if (!http || !dest || !dinfo || !size)
1114 {
1115 _cupsSetError(IPP_STATUS_ERROR_INTERNAL, strerror(EINVAL), 0);
1116 return (0);
1117 }
1118
1119 /*
1120 * Get the default media size, if any...
1121 */
1122
1123 if ((media = cupsGetOption("media", dest->num_options,
1124 dest->options)) == NULL)
1125 media = "na_letter_8.5x11in";
1126
1127 if (cupsGetDestMediaByName(http, dest, dinfo, media, flags, size))
1128 return (1);
1129
1130 if (strcmp(media, "na_letter_8.5x11in") &&
1131 cupsGetDestMediaByName(http, dest, dinfo, "iso_a4_210x297mm", flags,
1132 size))
1133 return (1);
1134
1135 if (strcmp(media, "iso_a4_210x297mm") &&
1136 cupsGetDestMediaByName(http, dest, dinfo, "na_letter_8.5x11in", flags,
1137 size))
1138 return (1);
1139
1140 if ((flags & CUPS_MEDIA_FLAGS_BORDERLESS) &&
1141 cupsGetDestMediaByName(http, dest, dinfo, "na_index_4x6in", flags, size))
1142 return (1);
1143
1144 /*
1145 * Fall back to the first matching media size...
1146 */
1147
1148 return (cupsGetDestMediaByIndex(http, dest, dinfo, 0, flags, size));
1149 }
1150
1151
1152 /*
1153 * 'cups_add_dconstres()' - Add a constraint or resolver to an array.
1154 */
1155
1156 static void
cups_add_dconstres(cups_array_t * a,ipp_t * collection)1157 cups_add_dconstres(
1158 cups_array_t *a, /* I - Array */
1159 ipp_t *collection) /* I - Collection value */
1160 {
1161 ipp_attribute_t *attr; /* Attribute */
1162 _cups_dconstres_t *temp; /* Current constraint/resolver */
1163
1164
1165 if ((attr = ippFindAttribute(collection, "resolver-name",
1166 IPP_TAG_NAME)) == NULL)
1167 return;
1168
1169 if ((temp = calloc(1, sizeof(_cups_dconstres_t))) == NULL)
1170 return;
1171
1172 temp->name = attr->values[0].string.text;
1173 temp->collection = collection;
1174
1175 cupsArrayAdd(a, temp);
1176 }
1177
1178
1179 /*
1180 * 'cups_compare_dconstres()' - Compare to resolver entries.
1181 */
1182
1183 static int /* O - Result of comparison */
cups_compare_dconstres(_cups_dconstres_t * a,_cups_dconstres_t * b)1184 cups_compare_dconstres(
1185 _cups_dconstres_t *a, /* I - First resolver */
1186 _cups_dconstres_t *b) /* I - Second resolver */
1187 {
1188 return (strcmp(a->name, b->name));
1189 }
1190
1191
1192 /*
1193 * 'cups_compare_media_db()' - Compare two media entries.
1194 */
1195
1196 static int /* O - Result of comparison */
cups_compare_media_db(_cups_media_db_t * a,_cups_media_db_t * b)1197 cups_compare_media_db(
1198 _cups_media_db_t *a, /* I - First media entries */
1199 _cups_media_db_t *b) /* I - Second media entries */
1200 {
1201 int result; /* Result of comparison */
1202
1203
1204 if ((result = a->width - b->width) == 0)
1205 result = a->length - b->length;
1206
1207 return (result);
1208 }
1209
1210
1211 /*
1212 * 'cups_copy_media_db()' - Copy a media entry.
1213 */
1214
1215 static _cups_media_db_t * /* O - New media entry */
cups_copy_media_db(_cups_media_db_t * mdb)1216 cups_copy_media_db(
1217 _cups_media_db_t *mdb) /* I - Media entry to copy */
1218 {
1219 _cups_media_db_t *temp; /* New media entry */
1220
1221
1222 if ((temp = calloc(1, sizeof(_cups_media_db_t))) == NULL)
1223 return (NULL);
1224
1225 if (mdb->color)
1226 temp->color = _cupsStrAlloc(mdb->color);
1227 if (mdb->key)
1228 temp->key = _cupsStrAlloc(mdb->key);
1229 if (mdb->info)
1230 temp->info = _cupsStrAlloc(mdb->info);
1231 if (mdb->size_name)
1232 temp->size_name = _cupsStrAlloc(mdb->size_name);
1233 if (mdb->source)
1234 temp->source = _cupsStrAlloc(mdb->source);
1235 if (mdb->type)
1236 temp->type = _cupsStrAlloc(mdb->type);
1237
1238 temp->width = mdb->width;
1239 temp->length = mdb->length;
1240 temp->bottom = mdb->bottom;
1241 temp->left = mdb->left;
1242 temp->right = mdb->right;
1243 temp->top = mdb->top;
1244
1245 return (temp);
1246 }
1247
1248
1249 /*
1250 * 'cups_create_cached()' - Create the media selection cache.
1251 */
1252
1253 static void
cups_create_cached(http_t * http,cups_dinfo_t * dinfo,unsigned flags)1254 cups_create_cached(http_t *http, /* I - Connection to destination */
1255 cups_dinfo_t *dinfo, /* I - Destination information */
1256 unsigned flags) /* I - Media selection flags */
1257 {
1258 cups_array_t *db; /* Media database array to use */
1259 _cups_media_db_t *mdb, /* Media database entry */
1260 *first; /* First entry this size */
1261
1262
1263 DEBUG_printf(("3cups_create_cached(http=%p, dinfo=%p, flags=%u)", (void *)http, (void *)dinfo, flags));
1264
1265 if (dinfo->cached_db)
1266 cupsArrayDelete(dinfo->cached_db);
1267
1268 dinfo->cached_db = cupsArrayNew(NULL, NULL);
1269 dinfo->cached_flags = flags;
1270
1271 if (flags & CUPS_MEDIA_FLAGS_READY)
1272 {
1273 DEBUG_puts("4cups_create_cached: ready media");
1274
1275 cups_update_ready(http, dinfo);
1276 db = dinfo->ready_db;
1277 }
1278 else
1279 {
1280 DEBUG_puts("4cups_create_cached: supported media");
1281
1282 if (!dinfo->media_db)
1283 cups_create_media_db(dinfo, CUPS_MEDIA_FLAGS_DEFAULT);
1284
1285 db = dinfo->media_db;
1286 }
1287
1288 for (mdb = (_cups_media_db_t *)cupsArrayFirst(db), first = mdb;
1289 mdb;
1290 mdb = (_cups_media_db_t *)cupsArrayNext(db))
1291 {
1292 DEBUG_printf(("4cups_create_cached: %p key=\"%s\", type=\"%s\", %dx%d, B%d L%d R%d T%d", (void *)mdb, mdb->key, mdb->type, mdb->width, mdb->length, mdb->bottom, mdb->left, mdb->right, mdb->top));
1293
1294 if (flags & CUPS_MEDIA_FLAGS_BORDERLESS)
1295 {
1296 if (!mdb->left && !mdb->right && !mdb->top && !mdb->bottom)
1297 {
1298 DEBUG_printf(("4cups_create_cached: add %p", (void *)mdb));
1299 cupsArrayAdd(dinfo->cached_db, mdb);
1300 }
1301 }
1302 else if (flags & CUPS_MEDIA_FLAGS_DUPLEX)
1303 {
1304 if (first->width != mdb->width || first->length != mdb->length)
1305 {
1306 DEBUG_printf(("4cups_create_cached: add %p", (void *)first));
1307 cupsArrayAdd(dinfo->cached_db, first);
1308 first = mdb;
1309 }
1310 else if (mdb->left >= first->left && mdb->right >= first->right && mdb->top >= first->top && mdb->bottom >= first->bottom &&
1311 (mdb->left != first->left || mdb->right != first->right || mdb->top != first->top || mdb->bottom != first->bottom))
1312 first = mdb;
1313 }
1314 else
1315 {
1316 DEBUG_printf(("4cups_create_cached: add %p", (void *)mdb));
1317 cupsArrayAdd(dinfo->cached_db, mdb);
1318 }
1319 }
1320
1321 if (flags & CUPS_MEDIA_FLAGS_DUPLEX)
1322 {
1323 DEBUG_printf(("4cups_create_cached: add %p", (void *)first));
1324 cupsArrayAdd(dinfo->cached_db, first);
1325 }
1326 }
1327
1328
1329 /*
1330 * 'cups_create_constraints()' - Create the constraints and resolvers arrays.
1331 */
1332
1333 static void
cups_create_constraints(cups_dinfo_t * dinfo)1334 cups_create_constraints(
1335 cups_dinfo_t *dinfo) /* I - Destination information */
1336 {
1337 int i; /* Looping var */
1338 ipp_attribute_t *attr; /* Attribute */
1339 _ipp_value_t *val; /* Current value */
1340
1341
1342 dinfo->constraints = cupsArrayNew3(NULL, NULL, NULL, 0, NULL,
1343 (cups_afree_func_t)free);
1344 dinfo->resolvers = cupsArrayNew3((cups_array_func_t)cups_compare_dconstres,
1345 NULL, NULL, 0, NULL,
1346 (cups_afree_func_t)free);
1347
1348 if ((attr = ippFindAttribute(dinfo->attrs, "job-constraints-supported",
1349 IPP_TAG_BEGIN_COLLECTION)) != NULL)
1350 {
1351 for (i = attr->num_values, val = attr->values; i > 0; i --, val ++)
1352 cups_add_dconstres(dinfo->constraints, val->collection);
1353 }
1354
1355 if ((attr = ippFindAttribute(dinfo->attrs, "job-resolvers-supported",
1356 IPP_TAG_BEGIN_COLLECTION)) != NULL)
1357 {
1358 for (i = attr->num_values, val = attr->values; i > 0; i --, val ++)
1359 cups_add_dconstres(dinfo->resolvers, val->collection);
1360 }
1361 }
1362
1363
1364 /*
1365 * 'cups_create_defaults()' - Create the -default option array.
1366 *
1367 * TODO: Need to support collection defaults...
1368 */
1369
1370 static void
cups_create_defaults(cups_dinfo_t * dinfo)1371 cups_create_defaults(
1372 cups_dinfo_t *dinfo) /* I - Destination information */
1373 {
1374 ipp_attribute_t *attr; /* Current attribute */
1375 char name[IPP_MAX_NAME + 1],
1376 /* Current name */
1377 *nameptr, /* Pointer into current name */
1378 value[2048]; /* Current value */
1379
1380
1381 /*
1382 * Iterate through the printer attributes looking for xxx-default and adding
1383 * xxx=value to the defaults option array.
1384 */
1385
1386 for (attr = ippFirstAttribute(dinfo->attrs);
1387 attr;
1388 attr = ippNextAttribute(dinfo->attrs))
1389 {
1390 if (!attr->name || attr->group_tag != IPP_TAG_PRINTER)
1391 continue;
1392
1393 if (attr->value_tag == IPP_TAG_BEGIN_COLLECTION)
1394 continue; /* TODO: STR #4096 */
1395
1396 if ((nameptr = attr->name + strlen(attr->name) - 8) <= attr->name ||
1397 strcmp(nameptr, "-default"))
1398 continue;
1399
1400 strlcpy(name, attr->name, sizeof(name));
1401 if ((nameptr = name + strlen(name) - 8) <= name ||
1402 strcmp(nameptr, "-default"))
1403 continue;
1404
1405 *nameptr = '\0';
1406
1407 if (ippAttributeString(attr, value, sizeof(value)) >= sizeof(value))
1408 continue;
1409
1410 dinfo->num_defaults = cupsAddOption(name, value, dinfo->num_defaults,
1411 &dinfo->defaults);
1412 }
1413 }
1414
1415
1416 /*
1417 * 'cups_create_media_db()' - Create the media database.
1418 */
1419
1420 static void
cups_create_media_db(cups_dinfo_t * dinfo,unsigned flags)1421 cups_create_media_db(
1422 cups_dinfo_t *dinfo, /* I - Destination information */
1423 unsigned flags) /* I - Media flags */
1424 {
1425 int i; /* Looping var */
1426 _ipp_value_t *val; /* Current value */
1427 ipp_attribute_t *media_col_db, /* media-col-database */
1428 *media_attr, /* media-xxx */
1429 *x_dimension, /* x-dimension */
1430 *y_dimension; /* y-dimension */
1431 pwg_media_t *pwg; /* PWG media info */
1432 cups_array_t *db; /* New media database array */
1433 _cups_media_db_t mdb; /* Media entry */
1434
1435
1436 db = cupsArrayNew3((cups_array_func_t)cups_compare_media_db,
1437 NULL, NULL, 0,
1438 (cups_acopy_func_t)cups_copy_media_db,
1439 (cups_afree_func_t)cups_free_media_db);
1440
1441 if (flags == CUPS_MEDIA_FLAGS_READY)
1442 {
1443 dinfo->ready_db = db;
1444
1445 media_col_db = ippFindAttribute(dinfo->ready_attrs, "media-col-ready",
1446 IPP_TAG_BEGIN_COLLECTION);
1447 media_attr = ippFindAttribute(dinfo->ready_attrs, "media-ready",
1448 IPP_TAG_ZERO);
1449 }
1450 else
1451 {
1452 dinfo->media_db = db;
1453 dinfo->min_size.width = INT_MAX;
1454 dinfo->min_size.length = INT_MAX;
1455 dinfo->max_size.width = 0;
1456 dinfo->max_size.length = 0;
1457
1458 media_col_db = ippFindAttribute(dinfo->attrs, "media-col-database",
1459 IPP_TAG_BEGIN_COLLECTION);
1460 media_attr = ippFindAttribute(dinfo->attrs, "media-supported",
1461 IPP_TAG_ZERO);
1462 }
1463
1464 if (media_col_db)
1465 {
1466 _ipp_value_t *custom = NULL; /* Custom size range value */
1467
1468 for (i = media_col_db->num_values, val = media_col_db->values;
1469 i > 0;
1470 i --, val ++)
1471 {
1472 memset(&mdb, 0, sizeof(mdb));
1473
1474 if ((media_attr = ippFindAttribute(val->collection, "media-size",
1475 IPP_TAG_BEGIN_COLLECTION)) != NULL)
1476 {
1477 ipp_t *media_size = media_attr->values[0].collection;
1478 /* media-size collection value */
1479
1480 if ((x_dimension = ippFindAttribute(media_size, "x-dimension",
1481 IPP_TAG_INTEGER)) != NULL &&
1482 (y_dimension = ippFindAttribute(media_size, "y-dimension",
1483 IPP_TAG_INTEGER)) != NULL)
1484 {
1485 /*
1486 * Fixed size...
1487 */
1488
1489 mdb.width = x_dimension->values[0].integer;
1490 mdb.length = y_dimension->values[0].integer;
1491 }
1492 else if ((x_dimension = ippFindAttribute(media_size, "x-dimension",
1493 IPP_TAG_INTEGER)) != NULL &&
1494 (y_dimension = ippFindAttribute(media_size, "y-dimension",
1495 IPP_TAG_RANGE)) != NULL)
1496 {
1497 /*
1498 * Roll limits...
1499 */
1500
1501 mdb.width = x_dimension->values[0].integer;
1502 mdb.length = y_dimension->values[0].range.upper;
1503 }
1504 else if (flags != CUPS_MEDIA_FLAGS_READY &&
1505 (x_dimension = ippFindAttribute(media_size, "x-dimension",
1506 IPP_TAG_RANGE)) != NULL &&
1507 (y_dimension = ippFindAttribute(media_size, "y-dimension",
1508 IPP_TAG_RANGE)) != NULL)
1509 {
1510 /*
1511 * Custom size range; save this as the custom size value with default
1512 * margins, then continue; we'll capture the real margins below...
1513 */
1514
1515 custom = val;
1516
1517 dinfo->min_size.width = x_dimension->values[0].range.lower;
1518 dinfo->min_size.length = y_dimension->values[0].range.lower;
1519 dinfo->min_size.left =
1520 dinfo->min_size.right = 635; /* Default 1/4" side margins */
1521 dinfo->min_size.top =
1522 dinfo->min_size.bottom = 1270; /* Default 1/2" top/bottom margins */
1523
1524 dinfo->max_size.width = x_dimension->values[0].range.upper;
1525 dinfo->max_size.length = y_dimension->values[0].range.upper;
1526 dinfo->max_size.left =
1527 dinfo->max_size.right = 635; /* Default 1/4" side margins */
1528 dinfo->max_size.top =
1529 dinfo->max_size.bottom = 1270; /* Default 1/2" top/bottom margins */
1530 continue;
1531 }
1532 }
1533
1534 if ((media_attr = ippFindAttribute(val->collection, "media-color",
1535 IPP_TAG_ZERO)) != NULL &&
1536 (media_attr->value_tag == IPP_TAG_NAME ||
1537 media_attr->value_tag == IPP_TAG_NAMELANG ||
1538 media_attr->value_tag == IPP_TAG_KEYWORD))
1539 mdb.color = media_attr->values[0].string.text;
1540
1541 if ((media_attr = ippFindAttribute(val->collection, "media-info",
1542 IPP_TAG_TEXT)) != NULL)
1543 mdb.info = media_attr->values[0].string.text;
1544
1545 if ((media_attr = ippFindAttribute(val->collection, "media-key",
1546 IPP_TAG_ZERO)) != NULL &&
1547 (media_attr->value_tag == IPP_TAG_NAME ||
1548 media_attr->value_tag == IPP_TAG_NAMELANG ||
1549 media_attr->value_tag == IPP_TAG_KEYWORD))
1550 mdb.key = media_attr->values[0].string.text;
1551
1552 if ((media_attr = ippFindAttribute(val->collection, "media-size-name",
1553 IPP_TAG_ZERO)) != NULL &&
1554 (media_attr->value_tag == IPP_TAG_NAME ||
1555 media_attr->value_tag == IPP_TAG_NAMELANG ||
1556 media_attr->value_tag == IPP_TAG_KEYWORD))
1557 mdb.size_name = media_attr->values[0].string.text;
1558
1559 if ((media_attr = ippFindAttribute(val->collection, "media-source",
1560 IPP_TAG_ZERO)) != NULL &&
1561 (media_attr->value_tag == IPP_TAG_NAME ||
1562 media_attr->value_tag == IPP_TAG_NAMELANG ||
1563 media_attr->value_tag == IPP_TAG_KEYWORD))
1564 mdb.source = media_attr->values[0].string.text;
1565
1566 if ((media_attr = ippFindAttribute(val->collection, "media-type",
1567 IPP_TAG_ZERO)) != NULL &&
1568 (media_attr->value_tag == IPP_TAG_NAME ||
1569 media_attr->value_tag == IPP_TAG_NAMELANG ||
1570 media_attr->value_tag == IPP_TAG_KEYWORD))
1571 mdb.type = media_attr->values[0].string.text;
1572
1573 if ((media_attr = ippFindAttribute(val->collection, "media-bottom-margin",
1574 IPP_TAG_INTEGER)) != NULL)
1575 mdb.bottom = media_attr->values[0].integer;
1576
1577 if ((media_attr = ippFindAttribute(val->collection, "media-left-margin",
1578 IPP_TAG_INTEGER)) != NULL)
1579 mdb.left = media_attr->values[0].integer;
1580
1581 if ((media_attr = ippFindAttribute(val->collection, "media-right-margin",
1582 IPP_TAG_INTEGER)) != NULL)
1583 mdb.right = media_attr->values[0].integer;
1584
1585 if ((media_attr = ippFindAttribute(val->collection, "media-top-margin",
1586 IPP_TAG_INTEGER)) != NULL)
1587 mdb.top = media_attr->values[0].integer;
1588
1589 cupsArrayAdd(db, &mdb);
1590 }
1591
1592 if (custom)
1593 {
1594 if ((media_attr = ippFindAttribute(custom->collection,
1595 "media-bottom-margin",
1596 IPP_TAG_INTEGER)) != NULL)
1597 {
1598 dinfo->min_size.top =
1599 dinfo->max_size.top = media_attr->values[0].integer;
1600 }
1601
1602 if ((media_attr = ippFindAttribute(custom->collection,
1603 "media-left-margin",
1604 IPP_TAG_INTEGER)) != NULL)
1605 {
1606 dinfo->min_size.left =
1607 dinfo->max_size.left = media_attr->values[0].integer;
1608 }
1609
1610 if ((media_attr = ippFindAttribute(custom->collection,
1611 "media-right-margin",
1612 IPP_TAG_INTEGER)) != NULL)
1613 {
1614 dinfo->min_size.right =
1615 dinfo->max_size.right = media_attr->values[0].integer;
1616 }
1617
1618 if ((media_attr = ippFindAttribute(custom->collection,
1619 "media-top-margin",
1620 IPP_TAG_INTEGER)) != NULL)
1621 {
1622 dinfo->min_size.top =
1623 dinfo->max_size.top = media_attr->values[0].integer;
1624 }
1625 }
1626 }
1627 else if (media_attr &&
1628 (media_attr->value_tag == IPP_TAG_NAME ||
1629 media_attr->value_tag == IPP_TAG_NAMELANG ||
1630 media_attr->value_tag == IPP_TAG_KEYWORD))
1631 {
1632 memset(&mdb, 0, sizeof(mdb));
1633
1634 mdb.left =
1635 mdb.right = 635; /* Default 1/4" side margins */
1636 mdb.top =
1637 mdb.bottom = 1270; /* Default 1/2" top/bottom margins */
1638
1639 for (i = media_attr->num_values, val = media_attr->values;
1640 i > 0;
1641 i --, val ++)
1642 {
1643 if ((pwg = pwgMediaForPWG(val->string.text)) == NULL)
1644 if ((pwg = pwgMediaForLegacy(val->string.text)) == NULL)
1645 {
1646 DEBUG_printf(("3cups_create_media_db: Ignoring unknown size '%s'.",
1647 val->string.text));
1648 continue;
1649 }
1650
1651 mdb.width = pwg->width;
1652 mdb.length = pwg->length;
1653
1654 if (flags != CUPS_MEDIA_FLAGS_READY &&
1655 !strncmp(val->string.text, "custom_min_", 11))
1656 {
1657 mdb.size_name = NULL;
1658 dinfo->min_size = mdb;
1659 }
1660 else if (flags != CUPS_MEDIA_FLAGS_READY &&
1661 !strncmp(val->string.text, "custom_max_", 11))
1662 {
1663 mdb.size_name = NULL;
1664 dinfo->max_size = mdb;
1665 }
1666 else
1667 {
1668 mdb.size_name = val->string.text;
1669
1670 cupsArrayAdd(db, &mdb);
1671 }
1672 }
1673 }
1674 }
1675
1676
1677 /*
1678 * 'cups_free_media_cb()' - Free a media entry.
1679 */
1680
1681 static void
cups_free_media_db(_cups_media_db_t * mdb)1682 cups_free_media_db(
1683 _cups_media_db_t *mdb) /* I - Media entry to free */
1684 {
1685 if (mdb->color)
1686 _cupsStrFree(mdb->color);
1687 if (mdb->key)
1688 _cupsStrFree(mdb->key);
1689 if (mdb->info)
1690 _cupsStrFree(mdb->info);
1691 if (mdb->size_name)
1692 _cupsStrFree(mdb->size_name);
1693 if (mdb->source)
1694 _cupsStrFree(mdb->source);
1695 if (mdb->type)
1696 _cupsStrFree(mdb->type);
1697
1698 free(mdb);
1699 }
1700
1701
1702 /*
1703 * 'cups_get_media_db()' - Lookup the media entry for a given size.
1704 */
1705
1706 static int /* O - 1 on match, 0 on failure */
cups_get_media_db(http_t * http,cups_dinfo_t * dinfo,pwg_media_t * pwg,unsigned flags,cups_size_t * size)1707 cups_get_media_db(http_t *http, /* I - Connection to destination */
1708 cups_dinfo_t *dinfo, /* I - Destination information */
1709 pwg_media_t *pwg, /* I - PWG media info */
1710 unsigned flags, /* I - Media matching flags */
1711 cups_size_t *size) /* O - Media size/margin/name info */
1712 {
1713 cups_array_t *db; /* Which media database to query */
1714 _cups_media_db_t *mdb, /* Current media database entry */
1715 *best = NULL, /* Best matching entry */
1716 key; /* Search key */
1717
1718
1719 /*
1720 * Create the media database as needed...
1721 */
1722
1723 if (flags & CUPS_MEDIA_FLAGS_READY)
1724 {
1725 cups_update_ready(http, dinfo);
1726 db = dinfo->ready_db;
1727 }
1728 else
1729 {
1730 if (!dinfo->media_db)
1731 cups_create_media_db(dinfo, CUPS_MEDIA_FLAGS_DEFAULT);
1732
1733 db = dinfo->media_db;
1734 }
1735
1736 /*
1737 * Find a match...
1738 */
1739
1740 memset(&key, 0, sizeof(key));
1741 key.width = pwg->width;
1742 key.length = pwg->length;
1743
1744 if ((mdb = cupsArrayFind(db, &key)) != NULL)
1745 {
1746 /*
1747 * Found an exact match, let's figure out the best margins for the flags
1748 * supplied...
1749 */
1750
1751 best = mdb;
1752
1753 if (flags & CUPS_MEDIA_FLAGS_BORDERLESS)
1754 {
1755 /*
1756 * Look for the smallest margins...
1757 */
1758
1759 if (best->left != 0 || best->right != 0 || best->top != 0 || best->bottom != 0)
1760 {
1761 for (mdb = (_cups_media_db_t *)cupsArrayNext(db);
1762 mdb && !cups_compare_media_db(mdb, &key);
1763 mdb = (_cups_media_db_t *)cupsArrayNext(db))
1764 {
1765 if (mdb->left <= best->left && mdb->right <= best->right &&
1766 mdb->top <= best->top && mdb->bottom <= best->bottom)
1767 {
1768 best = mdb;
1769 if (mdb->left == 0 && mdb->right == 0 && mdb->bottom == 0 &&
1770 mdb->top == 0)
1771 break;
1772 }
1773 }
1774 }
1775
1776 /*
1777 * If we need an exact match, return no-match if the size is not
1778 * borderless.
1779 */
1780
1781 if ((flags & CUPS_MEDIA_FLAGS_EXACT) &&
1782 (best->left || best->right || best->top || best->bottom))
1783 return (0);
1784 }
1785 else if (flags & CUPS_MEDIA_FLAGS_DUPLEX)
1786 {
1787 /*
1788 * Look for the largest margins...
1789 */
1790
1791 for (mdb = (_cups_media_db_t *)cupsArrayNext(db);
1792 mdb && !cups_compare_media_db(mdb, &key);
1793 mdb = (_cups_media_db_t *)cupsArrayNext(db))
1794 {
1795 if (mdb->left >= best->left && mdb->right >= best->right &&
1796 mdb->top >= best->top && mdb->bottom >= best->bottom &&
1797 (mdb->bottom != best->bottom || mdb->left != best->left || mdb->right != best->right || mdb->top != best->top))
1798 best = mdb;
1799 }
1800 }
1801 else
1802 {
1803 /*
1804 * Look for the smallest non-zero margins...
1805 */
1806
1807 for (mdb = (_cups_media_db_t *)cupsArrayNext(db);
1808 mdb && !cups_compare_media_db(mdb, &key);
1809 mdb = (_cups_media_db_t *)cupsArrayNext(db))
1810 {
1811 if (((mdb->left > 0 && mdb->left <= best->left) || best->left == 0) &&
1812 ((mdb->right > 0 && mdb->right <= best->right) || best->right == 0) &&
1813 ((mdb->top > 0 && mdb->top <= best->top) || best->top == 0) &&
1814 ((mdb->bottom > 0 && mdb->bottom <= best->bottom) || best->bottom == 0) &&
1815 (mdb->bottom != best->bottom || mdb->left != best->left || mdb->right != best->right || mdb->top != best->top))
1816 best = mdb;
1817 }
1818 }
1819 }
1820 else if (flags & CUPS_MEDIA_FLAGS_EXACT)
1821 {
1822 /*
1823 * See if we can do this as a custom size...
1824 */
1825
1826 if (pwg->width < dinfo->min_size.width ||
1827 pwg->width > dinfo->max_size.width ||
1828 pwg->length < dinfo->min_size.length ||
1829 pwg->length > dinfo->max_size.length)
1830 return (0); /* Out of range */
1831
1832 if ((flags & CUPS_MEDIA_FLAGS_BORDERLESS) &&
1833 (dinfo->min_size.left > 0 || dinfo->min_size.right > 0 ||
1834 dinfo->min_size.top > 0 || dinfo->min_size.bottom > 0))
1835 return (0); /* Not borderless */
1836
1837 key.size_name = (char *)pwg->pwg;
1838 key.bottom = dinfo->min_size.bottom;
1839 key.left = dinfo->min_size.left;
1840 key.right = dinfo->min_size.right;
1841 key.top = dinfo->min_size.top;
1842
1843 best = &key;
1844 }
1845 else if (pwg->width >= dinfo->min_size.width &&
1846 pwg->width <= dinfo->max_size.width &&
1847 pwg->length >= dinfo->min_size.length &&
1848 pwg->length <= dinfo->max_size.length)
1849 {
1850 /*
1851 * Map to custom size...
1852 */
1853
1854 key.size_name = (char *)pwg->pwg;
1855 key.bottom = dinfo->min_size.bottom;
1856 key.left = dinfo->min_size.left;
1857 key.right = dinfo->min_size.right;
1858 key.top = dinfo->min_size.top;
1859
1860 best = &key;
1861 }
1862 else
1863 {
1864 /*
1865 * Find a close size...
1866 */
1867
1868 for (mdb = (_cups_media_db_t *)cupsArrayFirst(db);
1869 mdb;
1870 mdb = (_cups_media_db_t *)cupsArrayNext(db))
1871 if (cups_is_close_media_db(mdb, &key))
1872 break;
1873
1874 if (!mdb)
1875 return (0);
1876
1877 best = mdb;
1878
1879 if (flags & CUPS_MEDIA_FLAGS_BORDERLESS)
1880 {
1881 /*
1882 * Look for the smallest margins...
1883 */
1884
1885 if (best->left != 0 || best->right != 0 || best->top != 0 ||
1886 best->bottom != 0)
1887 {
1888 for (mdb = (_cups_media_db_t *)cupsArrayNext(db);
1889 mdb && cups_is_close_media_db(mdb, &key);
1890 mdb = (_cups_media_db_t *)cupsArrayNext(db))
1891 {
1892 if (mdb->left <= best->left && mdb->right <= best->right &&
1893 mdb->top <= best->top && mdb->bottom <= best->bottom &&
1894 (mdb->bottom != best->bottom || mdb->left != best->left || mdb->right != best->right || mdb->top != best->top))
1895 {
1896 best = mdb;
1897 if (mdb->left == 0 && mdb->right == 0 && mdb->bottom == 0 &&
1898 mdb->top == 0)
1899 break;
1900 }
1901 }
1902 }
1903 }
1904 else if (flags & CUPS_MEDIA_FLAGS_DUPLEX)
1905 {
1906 /*
1907 * Look for the largest margins...
1908 */
1909
1910 for (mdb = (_cups_media_db_t *)cupsArrayNext(db);
1911 mdb && cups_is_close_media_db(mdb, &key);
1912 mdb = (_cups_media_db_t *)cupsArrayNext(db))
1913 {
1914 if (mdb->left >= best->left && mdb->right >= best->right &&
1915 mdb->top >= best->top && mdb->bottom >= best->bottom &&
1916 (mdb->bottom != best->bottom || mdb->left != best->left || mdb->right != best->right || mdb->top != best->top))
1917 best = mdb;
1918 }
1919 }
1920 else
1921 {
1922 /*
1923 * Look for the smallest non-zero margins...
1924 */
1925
1926 for (mdb = (_cups_media_db_t *)cupsArrayNext(db);
1927 mdb && cups_is_close_media_db(mdb, &key);
1928 mdb = (_cups_media_db_t *)cupsArrayNext(db))
1929 {
1930 if (((mdb->left > 0 && mdb->left <= best->left) || best->left == 0) &&
1931 ((mdb->right > 0 && mdb->right <= best->right) ||
1932 best->right == 0) &&
1933 ((mdb->top > 0 && mdb->top <= best->top) || best->top == 0) &&
1934 ((mdb->bottom > 0 && mdb->bottom <= best->bottom) ||
1935 best->bottom == 0) &&
1936 (mdb->bottom != best->bottom || mdb->left != best->left || mdb->right != best->right || mdb->top != best->top))
1937 best = mdb;
1938 }
1939 }
1940 }
1941
1942 if (best)
1943 {
1944 /*
1945 * Return the matching size...
1946 */
1947
1948 if (best->size_name)
1949 strlcpy(size->media, best->size_name, sizeof(size->media));
1950 else if (best->key)
1951 strlcpy(size->media, best->key, sizeof(size->media));
1952 else
1953 strlcpy(size->media, pwg->pwg, sizeof(size->media));
1954
1955 size->width = best->width;
1956 size->length = best->length;
1957 size->bottom = best->bottom;
1958 size->left = best->left;
1959 size->right = best->right;
1960 size->top = best->top;
1961
1962 return (1);
1963 }
1964
1965 return (0);
1966 }
1967
1968
1969 /*
1970 * 'cups_is_close_media_db()' - Compare two media entries to see if they are
1971 * close to the same size.
1972 *
1973 * Currently we use 5 points (from PostScript) as the matching range...
1974 */
1975
1976 static int /* O - 1 if the sizes are close */
cups_is_close_media_db(_cups_media_db_t * a,_cups_media_db_t * b)1977 cups_is_close_media_db(
1978 _cups_media_db_t *a, /* I - First media entries */
1979 _cups_media_db_t *b) /* I - Second media entries */
1980 {
1981 int dwidth, /* Difference in width */
1982 dlength; /* Difference in length */
1983
1984
1985 dwidth = a->width - b->width;
1986 dlength = a->length - b->length;
1987
1988 return (dwidth >= -176 && dwidth <= 176 &&
1989 dlength >= -176 && dlength <= 176);
1990 }
1991
1992
1993 /*
1994 * 'cups_test_constraints()' - Test constraints.
1995 *
1996 * TODO: STR #4096 - Need to properly support media-col contraints...
1997 */
1998
1999 static cups_array_t * /* O - Active constraints */
cups_test_constraints(cups_dinfo_t * dinfo,const char * new_option,const char * new_value,int num_options,cups_option_t * options,int * num_conflicts,cups_option_t ** conflicts)2000 cups_test_constraints(
2001 cups_dinfo_t *dinfo, /* I - Destination information */
2002 const char *new_option, /* I - Newly selected option */
2003 const char *new_value, /* I - Newly selected value */
2004 int num_options, /* I - Number of options */
2005 cups_option_t *options, /* I - Options */
2006 int *num_conflicts, /* O - Number of conflicting options */
2007 cups_option_t **conflicts) /* O - Conflicting options */
2008 {
2009 int i, /* Looping var */
2010 match; /* Value matches? */
2011 int num_matching; /* Number of matching options */
2012 cups_option_t *matching; /* Matching options */
2013 _cups_dconstres_t *c; /* Current constraint */
2014 cups_array_t *active = NULL; /* Active constraints */
2015 ipp_attribute_t *attr; /* Current attribute */
2016 _ipp_value_t *attrval; /* Current attribute value */
2017 const char *value; /* Current value */
2018 char temp[1024]; /* Temporary string */
2019 int int_value; /* Integer value */
2020 int xres_value, /* Horizontal resolution */
2021 yres_value; /* Vertical resolution */
2022 ipp_res_t units_value; /* Resolution units */
2023
2024
2025 for (c = (_cups_dconstres_t *)cupsArrayFirst(dinfo->constraints);
2026 c;
2027 c = (_cups_dconstres_t *)cupsArrayNext(dinfo->constraints))
2028 {
2029 num_matching = 0;
2030 matching = NULL;
2031
2032 for (attr = ippFirstAttribute(c->collection);
2033 attr;
2034 attr = ippNextAttribute(c->collection))
2035 {
2036 if (attr->value_tag == IPP_TAG_BEGIN_COLLECTION)
2037 break; /* TODO: STR #4096 */
2038
2039 /*
2040 * Get the value for the current attribute in the constraint...
2041 */
2042
2043 if (new_option && new_value && !strcmp(attr->name, new_option))
2044 value = new_value;
2045 else if ((value = cupsGetOption(attr->name, num_options,
2046 options)) == NULL)
2047 value = cupsGetOption(attr->name, dinfo->num_defaults, dinfo->defaults);
2048
2049 if (!value)
2050 {
2051 /*
2052 * Not set so this constraint does not apply...
2053 */
2054
2055 break;
2056 }
2057
2058 match = 0;
2059
2060 switch (attr->value_tag)
2061 {
2062 case IPP_TAG_INTEGER :
2063 case IPP_TAG_ENUM :
2064 int_value = atoi(value);
2065
2066 for (i = attr->num_values, attrval = attr->values;
2067 i > 0;
2068 i --, attrval ++)
2069 {
2070 if (attrval->integer == int_value)
2071 {
2072 match = 1;
2073 break;
2074 }
2075 }
2076 break;
2077
2078 case IPP_TAG_BOOLEAN :
2079 int_value = !strcmp(value, "true");
2080
2081 for (i = attr->num_values, attrval = attr->values;
2082 i > 0;
2083 i --, attrval ++)
2084 {
2085 if (attrval->boolean == int_value)
2086 {
2087 match = 1;
2088 break;
2089 }
2090 }
2091 break;
2092
2093 case IPP_TAG_RANGE :
2094 int_value = atoi(value);
2095
2096 for (i = attr->num_values, attrval = attr->values;
2097 i > 0;
2098 i --, attrval ++)
2099 {
2100 if (int_value >= attrval->range.lower &&
2101 int_value <= attrval->range.upper)
2102 {
2103 match = 1;
2104 break;
2105 }
2106 }
2107 break;
2108
2109 case IPP_TAG_RESOLUTION :
2110 if (sscanf(value, "%dx%d%15s", &xres_value, &yres_value, temp) != 3)
2111 {
2112 if (sscanf(value, "%d%15s", &xres_value, temp) != 2)
2113 break;
2114
2115 yres_value = xres_value;
2116 }
2117
2118 if (!strcmp(temp, "dpi"))
2119 units_value = IPP_RES_PER_INCH;
2120 else if (!strcmp(temp, "dpc") || !strcmp(temp, "dpcm"))
2121 units_value = IPP_RES_PER_CM;
2122 else
2123 break;
2124
2125 for (i = attr->num_values, attrval = attr->values;
2126 i > 0;
2127 i --, attrval ++)
2128 {
2129 if (attrval->resolution.xres == xres_value &&
2130 attrval->resolution.yres == yres_value &&
2131 attrval->resolution.units == units_value)
2132 {
2133 match = 1;
2134 break;
2135 }
2136 }
2137 break;
2138
2139 case IPP_TAG_TEXT :
2140 case IPP_TAG_NAME :
2141 case IPP_TAG_KEYWORD :
2142 case IPP_TAG_CHARSET :
2143 case IPP_TAG_URI :
2144 case IPP_TAG_URISCHEME :
2145 case IPP_TAG_MIMETYPE :
2146 case IPP_TAG_LANGUAGE :
2147 case IPP_TAG_TEXTLANG :
2148 case IPP_TAG_NAMELANG :
2149 for (i = attr->num_values, attrval = attr->values;
2150 i > 0;
2151 i --, attrval ++)
2152 {
2153 if (!strcmp(attrval->string.text, value))
2154 {
2155 match = 1;
2156 break;
2157 }
2158 }
2159 break;
2160
2161 default :
2162 break;
2163 }
2164
2165 if (!match)
2166 break;
2167
2168 num_matching = cupsAddOption(attr->name, value, num_matching, &matching);
2169 }
2170
2171 if (!attr)
2172 {
2173 if (!active)
2174 active = cupsArrayNew(NULL, NULL);
2175
2176 cupsArrayAdd(active, c);
2177
2178 if (num_conflicts && conflicts)
2179 {
2180 cups_option_t *moption; /* Matching option */
2181
2182 for (i = num_matching, moption = matching; i > 0; i --, moption ++)
2183 *num_conflicts = cupsAddOption(moption->name, moption->value,
2184 *num_conflicts, conflicts);
2185 }
2186 }
2187
2188 cupsFreeOptions(num_matching, matching);
2189 }
2190
2191 return (active);
2192 }
2193
2194
2195 /*
2196 * 'cups_update_ready()' - Update xxx-ready attributes for the printer.
2197 */
2198
2199 static void
cups_update_ready(http_t * http,cups_dinfo_t * dinfo)2200 cups_update_ready(http_t *http, /* I - Connection to destination */
2201 cups_dinfo_t *dinfo) /* I - Destination information */
2202 {
2203 ipp_t *request; /* Get-Printer-Attributes request */
2204 static const char * const pattrs[] = /* Printer attributes we want */
2205 {
2206 "finishings-col-ready",
2207 "finishings-ready",
2208 "job-finishings-col-ready",
2209 "job-finishings-ready",
2210 "media-col-ready",
2211 "media-ready"
2212 };
2213
2214
2215 /*
2216 * Don't update more than once every 30 seconds...
2217 */
2218
2219 if ((time(NULL) - dinfo->ready_time) < _CUPS_MEDIA_READY_TTL)
2220 return;
2221
2222 /*
2223 * Free any previous results...
2224 */
2225
2226 if (dinfo->cached_flags & CUPS_MEDIA_FLAGS_READY)
2227 {
2228 cupsArrayDelete(dinfo->cached_db);
2229 dinfo->cached_db = NULL;
2230 dinfo->cached_flags = CUPS_MEDIA_FLAGS_DEFAULT;
2231 }
2232
2233 ippDelete(dinfo->ready_attrs);
2234 dinfo->ready_attrs = NULL;
2235
2236 cupsArrayDelete(dinfo->ready_db);
2237 dinfo->ready_db = NULL;
2238
2239 /*
2240 * Query the xxx-ready values...
2241 */
2242
2243 request = ippNewRequest(IPP_OP_GET_PRINTER_ATTRIBUTES);
2244 ippSetVersion(request, dinfo->version / 10, dinfo->version % 10);
2245
2246 ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", NULL,
2247 dinfo->uri);
2248 ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "requesting-user-name",
2249 NULL, cupsUser());
2250 ippAddStrings(request, IPP_TAG_OPERATION, IPP_CONST_TAG(IPP_TAG_KEYWORD), "requested-attributes", (int)(sizeof(pattrs) / sizeof(pattrs[0])), NULL, pattrs);
2251
2252 dinfo->ready_attrs = cupsDoRequest(http, request, dinfo->resource);
2253
2254 /*
2255 * Update the ready media database...
2256 */
2257
2258 cups_create_media_db(dinfo, CUPS_MEDIA_FLAGS_READY);
2259
2260 /*
2261 * Update last lookup time and return...
2262 */
2263
2264 dinfo->ready_time = time(NULL);
2265 }
2266